@wise/wds-codemods 1.2.1-experimental-18899d8 → 1.2.1-experimental-153ecf4

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 CHANGED
@@ -34,6 +34,7 @@ const getDistinctId = () => {
34
34
  */
35
35
  const toMixpanelProperties = (properties) => ({
36
36
  Transform: properties.transform,
37
+ Stage: properties.stage,
37
38
  Engine: properties.engine,
38
39
  "Is Monorepo": properties.isMonorepo,
39
40
  "Is Git Ignore Enabled": properties.useGitIgnore,
@@ -50,20 +51,17 @@ const toMixpanelProperties = (properties) => ({
50
51
  * Event names follow Wise Mixpanel naming conventions:
51
52
  * `[Subject] - [Action]` with past-tense verbs, capitalized words, delimited by " - ".
52
53
  */
53
- const TRACKING_EVENTS = {
54
- STARTED: "WDS - Codemod - Started",
55
- FINISHED: "WDS - Codemod - Finished",
56
- FAILED: "WDS - Codemod - Failed"
57
- };
54
+ const TRACKING_EVENT = "WDS - Codemod";
58
55
  /**
59
56
  * Fire-and-forget event tracking. Never throws — failures are silently ignored
60
57
  * so tracking never interferes with the CLI experience.
61
58
  */
62
59
  const track = (event, properties) => {
63
60
  try {
64
- const mp = getClient();
65
61
  const mapped = toMixpanelProperties(properties);
66
62
  if (properties.isDebug) console.debug(`${CONSOLE_ICONS.info} [Mixpanel] ${event}`, JSON.stringify(mapped, null, 2));
63
+ if (properties.noTracking) return;
64
+ const mp = getClient();
67
65
  if (!mp) return;
68
66
  mp.track(event, {
69
67
  distinct_id: getDistinctId(),
@@ -138,6 +136,7 @@ async function runCodemod(transformsDir) {
138
136
  const args = process.argv.slice(2);
139
137
  const candidate = args[0];
140
138
  isDebug = args.includes("--debug");
139
+ const noTracking = args.includes("--no-tracking");
141
140
  const startTime = Date.now();
142
141
  try {
143
142
  const root = findProjectRoot();
@@ -188,9 +187,13 @@ async function runCodemod(transformsDir) {
188
187
  selectionMethod,
189
188
  targetPathCount: options.targetPaths.length,
190
189
  isDebug,
190
+ noTracking,
191
191
  repository: path.basename(root)
192
192
  };
193
- track(TRACKING_EVENTS.STARTED, trackingProps);
193
+ track(TRACKING_EVENT, {
194
+ ...trackingProps,
195
+ stage: "Started"
196
+ });
194
197
  if (codemodConfig?.type === "claude") {
195
198
  handleClaudePrereqs({
196
199
  options,
@@ -220,15 +223,17 @@ async function runCodemod(transformsDir) {
220
223
  }));
221
224
  }
222
225
  if (codemodConfig?.type === "jscodeshift") await summariseReportFile(reportPath);
223
- track(TRACKING_EVENTS.FINISHED, {
226
+ track(TRACKING_EVENT, {
224
227
  ...trackingProps,
228
+ stage: "Finished",
225
229
  durationMs: Date.now() - startTime
226
230
  });
227
231
  } catch (error) {
228
232
  if (error instanceof Error) console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, error.message);
229
233
  else console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, String(error));
230
- track(TRACKING_EVENTS.FAILED, {
234
+ track(TRACKING_EVENT, {
231
235
  transform: candidate ?? "unknown",
236
+ stage: "Failed",
232
237
  error: error instanceof Error ? error.message : String(error),
233
238
  durationMs: Date.now() - startTime
234
239
  });
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["list"],"sources":["../src/helpers/tracking/index.ts","../src/controller/helpers/claudePrereqs.ts","../src/controller/index.ts","../src/index.ts"],"sourcesContent":["import { createHash } from 'node:crypto';\nimport { hostname, userInfo } from 'node:os';\n\nimport Mixpanel from 'mixpanel';\n\nimport { CONSOLE_ICONS } from '../../constants/common';\n\nconst MIXPANEL_TOKEN = '8ba4a7a5182f05e0a79ded57d5d2f051';\n\nlet client: Mixpanel.Mixpanel | null = null;\n\nconst getClient = (): Mixpanel.Mixpanel | null => {\n if (!MIXPANEL_TOKEN) return null;\n client ??= Mixpanel.init(MIXPANEL_TOKEN, {\n geolocate: false,\n });\n return client;\n};\n\n/**\n * Anonymous but stable user identifier derived from OS username + hostname.\n * No PII is sent — only a SHA-256 hash.\n */\nconst getDistinctId = (): string => {\n const raw = `${userInfo().username}@${hostname()}`;\n return createHash('sha256').update(raw).digest('hex').slice(0, 16);\n};\n\n/**\n * Internal interface for passing tracking data from call sites.\n * Property names are mapped to Wise Mixpanel naming conventions before sending.\n *\n * @see https://transferwise.atlassian.net/wiki — Handbook for Mixpanel tracking\n */\nexport interface TrackingProperties {\n transform: string;\n engine?: 'jscodeshift' | 'claude';\n isMonorepo?: boolean;\n useGitIgnore?: boolean;\n hasIgnorePatterns?: boolean;\n selectionMethod?: 'cli' | 'interactive';\n targetPathCount?: number;\n isDebug?: boolean;\n repository?: string;\n error?: string;\n durationMs?: number;\n}\n\n/**\n * Maps internal camelCase properties to Wise Mixpanel naming conventions:\n * - Spaces between words, each word capitalized\n * - Booleans prefixed with \"Is\"\n * - Counts use \"Number Of\"\n * - Duration in seconds (not ms)\n */\nconst toMixpanelProperties = (\n properties: TrackingProperties,\n): Record<string, string | number | boolean | undefined> => ({\n Transform: properties.transform,\n Engine: properties.engine,\n 'Is Monorepo': properties.isMonorepo,\n 'Is Git Ignore Enabled': properties.useGitIgnore,\n 'Is Ignore Patterns Used': properties.hasIgnorePatterns,\n 'Selection Method': properties.selectionMethod,\n 'Number Of Target Paths': properties.targetPathCount,\n 'Is Debug': properties.isDebug,\n Repository: properties.repository,\n Timestamp: new Date().toISOString(),\n Error: properties.error,\n Duration: properties.durationMs != null ? properties.durationMs / 1000 : undefined,\n});\n\n/**\n * Event names follow Wise Mixpanel naming conventions:\n * `[Subject] - [Action]` with past-tense verbs, capitalized words, delimited by \" - \".\n */\nexport const TRACKING_EVENTS = {\n STARTED: 'WDS - Codemod - Started',\n FINISHED: 'WDS - Codemod - Finished',\n FAILED: 'WDS - Codemod - Failed',\n} as const;\n\n/**\n * Fire-and-forget event tracking. Never throws — failures are silently ignored\n * so tracking never interferes with the CLI experience.\n */\nexport const track = (event: string, properties: TrackingProperties): void => {\n try {\n const mp = getClient();\n const mapped = toMixpanelProperties(properties);\n\n if (properties.isDebug) {\n console.debug(`${CONSOLE_ICONS.info} [Mixpanel] ${event}`, JSON.stringify(mapped, null, 2));\n }\n\n if (!mp) return;\n\n mp.track(event, {\n distinct_id: getDistinctId(),\n ...mapped,\n });\n } catch {\n // Intentionally swallowed — tracking must never break the CLI\n }\n};\n\n/**\n * Returns a flush promise you can await before process.exit.\n * Mixpanel's Node client batches requests, so we give it a small window.\n */\nexport const flushTracking = async (): Promise<void> =>\n new Promise((resolve) => {\n // Mixpanel Node SDK sends HTTP requests async. Give it a brief grace period.\n setTimeout(resolve, 300);\n });\n","import type Spinnies from 'spinnies';\n\nimport { CONSOLE_ICONS } from '../../constants/common';\nimport type { CodemodOptions } from '../types';\nimport { assessPrerequisitesBatch } from './dependencyChecks';\n\ninterface ClaudePrereqHandlerOptions {\n options: CodemodOptions;\n spinners: Spinnies;\n codemodPath: string;\n}\n\nexport function handleClaudePrereqs({\n options,\n codemodPath,\n spinners,\n}: ClaudePrereqHandlerOptions) {\n spinners.add('prerequisite-check', { text: 'Checking prerequisites...' });\n // Fail-fast: check prerequisites for all target paths up-front\n const { allPassed, packageRootToTargets, failedPackageRoots } = assessPrerequisitesBatch(\n options.targetPaths,\n codemodPath,\n spinners,\n );\n if (!allPassed) {\n spinners.add('prerequisite-partial-failure', {\n text: 'One or more packages failed prerequisite checks - these targets will be skipped:',\n status: 'fail',\n });\n\n for (const failedRoot of failedPackageRoots) {\n const targets = packageRootToTargets.get(failedRoot) ?? [];\n // Filter out targets that belong to failed package roots\n // eslint-disable-next-line no-param-reassign\n options.targetPaths = options.targetPaths.filter(\n (targetPath) => !targets.includes(targetPath),\n );\n spinners.add(`prerequisite-fail-${failedRoot}`, {\n text: `- \\x1b[2m\\x1b[31m${failedRoot}\\x1b[0m${targets.length > 0 && targets[0] !== failedRoot ? ` (targets: ${targets.join(', ')})` : ''}\\x1b[0m`,\n indent: 2,\n status: 'non-spinnable',\n });\n }\n }\n\n if (!options.targetPaths.length) {\n spinners.add('no-valid-targets', {\n text: `${CONSOLE_ICONS.error} No valid target paths remaining after prerequisite checks - exiting.`,\n status: 'fail',\n });\n spinners.stopAll('fail');\n spinners.checkIfActiveSpinners();\n }\n}\n","#!/usr/bin/env node\n\nimport { execSync } from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nimport { select as list } from '@inquirer/prompts';\nimport Spinnies from 'spinnies';\n\nimport { CONSOLE_ICONS } from '../constants/common';\nimport {\n flushTracking,\n track,\n TRACKING_EVENTS,\n type TrackingProperties,\n} from '../helpers/tracking';\nimport {\n assessPrerequisites,\n findPackages,\n findProjectRoot,\n getCodemodConfig,\n getOptions,\n loadTransformModules,\n logToInquirer,\n runTransformPrompts,\n validateClaudeConfig,\n} from './helpers';\nimport { handleClaudePrereqs } from './helpers/claudePrereqs';\n\nlet isDebug = false;\nconst currentFilePath = fileURLToPath(import.meta.url);\nconst currentDirPath = path.dirname(currentFilePath);\n\nconst resetReportFile = async (reportPath: string) => {\n try {\n await fs.access(reportPath);\n await fs.rm(reportPath);\n console.debug(\n `${CONSOLE_ICONS.info} Removed existing report file${isDebug ? `: ${reportPath}` : '.'}`,\n );\n } catch {\n console.debug(\n `${CONSOLE_ICONS.info} No existing report file to remove${isDebug ? `: ${reportPath}` : '.'}`,\n );\n }\n};\n\nconst summariseReportFile = async (reportPath: string) => {\n try {\n const reportContent = await fs.readFile(reportPath, 'utf8');\n const lines = reportContent.split('\\n').filter(Boolean);\n if (lines.length) {\n console.debug(\n `\\n${CONSOLE_ICONS.warning} ${lines.length} manual review${lines.length > 1 ? 's are' : ' is'} required. See ${reportPath} for details.`,\n );\n } else {\n console.debug(\n `${CONSOLE_ICONS.info} Report file exists but is empty${isDebug ? `: ${reportPath}` : '.'}`,\n );\n }\n } catch {\n console.debug(`${CONSOLE_ICONS.info} No report file generated - no manual reviews needed`);\n }\n};\n\nconst log = (label: string, value?: string): void => {\n if (typeof logToInquirer === 'function') {\n logToInquirer(label, value || '');\n } else {\n console.info(label, value || '');\n }\n};\n\nasync function runCodemod(transformsDir?: string) {\n const args = process.argv.slice(2);\n const candidate = args[0];\n isDebug = args.includes('--debug');\n const startTime = Date.now();\n\n try {\n const root = findProjectRoot();\n const packagesPromise = findPackages();\n const resolvedTransformsDir =\n transformsDir ?? path.resolve(currentDirPath, '../dist/transforms');\n\n if (isDebug) {\n console.debug(\n `${CONSOLE_ICONS.info} Resolved transforms directory: ${resolvedTransformsDir}`,\n );\n }\n\n const { transformFiles: resolvedTransformNames } =\n await loadTransformModules(resolvedTransformsDir);\n if (resolvedTransformNames.length === 0) {\n throw new Error(\n `${CONSOLE_ICONS.error} No transform scripts found${isDebug ? ` in: ${resolvedTransformsDir}` : '.'}`,\n );\n }\n\n let transformFile: string;\n\n if (candidate && resolvedTransformNames.includes(candidate)) {\n log('Select codemod to run:', candidate);\n transformFile = candidate;\n } else {\n transformFile = await list({\n message: 'Select codemod to run:',\n choices: resolvedTransformNames.map((name: string) => ({ name, value: name })),\n });\n log('Selected codemod:', transformFile);\n }\n\n const codemodPath = path.resolve(resolvedTransformsDir, transformFile, 'transformer.js');\n const codemodConfig = getCodemodConfig(codemodPath);\n if (isDebug) {\n console.debug(`${CONSOLE_ICONS.info} Resolved codemod path: ${codemodPath}`);\n }\n\n if (codemodConfig?.type === 'claude') {\n validateClaudeConfig();\n }\n\n const spinners = new Spinnies();\n spinners.add('loading-codemod', { text: `Loading codemod (${transformFile})...` });\n const packages = await packagesPromise;\n const reportPath = path.resolve(root, 'codemod-report.txt');\n spinners.succeed('loading-codemod', {\n text: `Successfully loaded codemod: \\x1b[2m${transformFile}\\x1b[0m`,\n });\n\n if (codemodConfig?.type === 'jscodeshift') {\n await resetReportFile(reportPath);\n }\n\n const promptAnswers = await runTransformPrompts(codemodPath);\n const options = await getOptions({\n packages,\n root,\n transformFiles: resolvedTransformNames,\n preselectedTransformFile: transformFile,\n transformerType: codemodConfig?.type,\n });\n\n const selectionMethod: TrackingProperties['selectionMethod'] =\n candidate && resolvedTransformNames.includes(candidate) ? 'cli' : 'interactive';\n\n const trackingProps: TrackingProperties = {\n transform: transformFile,\n engine: codemodConfig?.type,\n isMonorepo: args.includes('--monorepo'),\n useGitIgnore: options.useGitIgnore,\n hasIgnorePatterns: Boolean(options.ignorePatterns),\n selectionMethod,\n targetPathCount: options.targetPaths.length,\n isDebug,\n repository: path.basename(root),\n };\n\n track(TRACKING_EVENTS.STARTED, trackingProps);\n\n // Handle Claude transforms differently, as they work on multiple targets at once\n if (codemodConfig?.type === 'claude') {\n handleClaudePrereqs({ options, codemodPath, spinners });\n\n // Dynamically import the transformer module\n const transformerModule = (await import(codemodPath)) as {\n default: (targetPaths: string[], isDebug?: boolean) => Promise<void>;\n };\n const transformer = transformerModule.default;\n await transformer(options.targetPaths, isDebug);\n } else {\n // Button codemod doesn't use spinnies\n spinners.stopAll('succeed');\n spinners.checkIfActiveSpinners();\n\n await Promise.all(\n options.targetPaths.map(async (targetPath) => {\n console.info(\n `${CONSOLE_ICONS.focus} \\x1b[1mProcessing:\\x1b[0m \\x1b[32m${targetPath}\\x1b[0m`,\n );\n\n // Check prerequisites for this target before running\n // TODO: re-enable after testing tracking\n // const ok = assessPrerequisites(targetPath, codemodPath);\n // if (!ok) {\n // return;\n // }\n\n const answerArgs = Object.entries(promptAnswers).map(\n ([promptName, answerValue]) => `--${promptName}=${String(answerValue)}`,\n );\n\n const argsList = [\n '-t',\n codemodPath,\n targetPath,\n options.isDry ? '--dry' : '',\n options.isPrint ? '--print' : '',\n options.ignorePatterns\n ? options.ignorePatterns\n .split(',')\n .map((pattern) => `--ignore-pattern=${pattern.trim()}`)\n .join(' ')\n : '',\n options.useGitIgnore ? '--gitignore' : '',\n ...answerArgs,\n ].filter(Boolean);\n const command = `npx jscodeshift ${argsList.join(' ')}`;\n\n if (isDebug) {\n console.debug(`${CONSOLE_ICONS.info} Running: ${command}`);\n }\n\n return execSync(command, { stdio: 'inherit' });\n }),\n );\n }\n\n if (codemodConfig?.type === 'jscodeshift') {\n await summariseReportFile(reportPath);\n }\n\n track(TRACKING_EVENTS.FINISHED, {\n ...trackingProps,\n durationMs: Date.now() - startTime,\n });\n } catch (error: unknown) {\n if (error instanceof Error) {\n console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, error.message);\n } else {\n console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, String(error));\n }\n track(TRACKING_EVENTS.FAILED, {\n transform: candidate ?? 'unknown',\n error: error instanceof Error ? error.message : String(error),\n durationMs: Date.now() - startTime,\n });\n\n await flushTracking();\n\n if (process.env.NODE_ENV !== 'test') {\n process.exit(1);\n }\n }\n\n await flushTracking();\n}\n\nexport { runCodemod };\n","#!/usr/bin/env node\nimport { runCodemod } from './controller';\n\nvoid runCodemod();\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["list"],"sources":["../src/helpers/tracking/index.ts","../src/controller/helpers/claudePrereqs.ts","../src/controller/index.ts","../src/index.ts"],"sourcesContent":["import { createHash } from 'node:crypto';\nimport { hostname, userInfo } from 'node:os';\n\nimport Mixpanel from 'mixpanel';\n\nimport { CONSOLE_ICONS } from '../../constants/common';\n\nconst MIXPANEL_TOKEN = '8ba4a7a5182f05e0a79ded57d5d2f051';\n\nlet client: Mixpanel.Mixpanel | null = null;\n\nconst getClient = (): Mixpanel.Mixpanel | null => {\n if (!MIXPANEL_TOKEN) return null;\n client ??= Mixpanel.init(MIXPANEL_TOKEN, {\n geolocate: false,\n });\n return client;\n};\n\n/**\n * Anonymous but stable user identifier derived from OS username + hostname.\n * No PII is sent — only a SHA-256 hash.\n */\nconst getDistinctId = (): string => {\n const raw = `${userInfo().username}@${hostname()}`;\n return createHash('sha256').update(raw).digest('hex').slice(0, 16);\n};\n\n/**\n * Internal interface for passing tracking data from call sites.\n * Property names are mapped to Wise Mixpanel naming conventions before sending.\n *\n * @see https://transferwise.atlassian.net/wiki — Handbook for Mixpanel tracking\n */\nexport interface TrackingProperties {\n transform: string;\n stage: 'Started' | 'Finished' | 'Failed';\n engine?: 'jscodeshift' | 'claude';\n isMonorepo?: boolean;\n useGitIgnore?: boolean;\n hasIgnorePatterns?: boolean;\n selectionMethod?: 'cli' | 'interactive';\n targetPathCount?: number;\n isDebug?: boolean;\n noTracking?: boolean;\n repository?: string;\n error?: string;\n durationMs?: number;\n}\n\n/**\n * Maps internal camelCase properties to Wise Mixpanel naming conventions:\n * - Spaces between words, each word capitalized\n * - Booleans prefixed with \"Is\"\n * - Counts use \"Number Of\"\n * - Duration in seconds (not ms)\n */\nconst toMixpanelProperties = (\n properties: TrackingProperties,\n): Record<string, string | number | boolean | undefined> => ({\n Transform: properties.transform,\n Stage: properties.stage,\n Engine: properties.engine,\n 'Is Monorepo': properties.isMonorepo,\n 'Is Git Ignore Enabled': properties.useGitIgnore,\n 'Is Ignore Patterns Used': properties.hasIgnorePatterns,\n 'Selection Method': properties.selectionMethod,\n 'Number Of Target Paths': properties.targetPathCount,\n 'Is Debug': properties.isDebug,\n Repository: properties.repository,\n Timestamp: new Date().toISOString(),\n Error: properties.error,\n Duration: properties.durationMs != null ? properties.durationMs / 1000 : undefined,\n});\n\n/**\n * Event names follow Wise Mixpanel naming conventions:\n * `[Subject] - [Action]` with past-tense verbs, capitalized words, delimited by \" - \".\n */\nexport const TRACKING_EVENT = 'WDS - Codemod' as const;\n\n/**\n * Fire-and-forget event tracking. Never throws — failures are silently ignored\n * so tracking never interferes with the CLI experience.\n */\nexport const track = (event: string, properties: TrackingProperties): void => {\n try {\n const mapped = toMixpanelProperties(properties);\n\n if (properties.isDebug) {\n console.debug(`${CONSOLE_ICONS.info} [Mixpanel] ${event}`, JSON.stringify(mapped, null, 2));\n }\n\n if (properties.noTracking) return;\n\n const mp = getClient();\n if (!mp) return;\n\n mp.track(event, {\n distinct_id: getDistinctId(),\n ...mapped,\n });\n } catch {\n // Intentionally swallowed — tracking must never break the CLI\n }\n};\n\n/**\n * Returns a flush promise you can await before process.exit.\n * Mixpanel's Node client batches requests, so we give it a small window.\n */\nexport const flushTracking = async (): Promise<void> =>\n new Promise((resolve) => {\n // Mixpanel Node SDK sends HTTP requests async. Give it a brief grace period.\n setTimeout(resolve, 300);\n });\n","import type Spinnies from 'spinnies';\n\nimport { CONSOLE_ICONS } from '../../constants/common';\nimport type { CodemodOptions } from '../types';\nimport { assessPrerequisitesBatch } from './dependencyChecks';\n\ninterface ClaudePrereqHandlerOptions {\n options: CodemodOptions;\n spinners: Spinnies;\n codemodPath: string;\n}\n\nexport function handleClaudePrereqs({\n options,\n codemodPath,\n spinners,\n}: ClaudePrereqHandlerOptions) {\n spinners.add('prerequisite-check', { text: 'Checking prerequisites...' });\n // Fail-fast: check prerequisites for all target paths up-front\n const { allPassed, packageRootToTargets, failedPackageRoots } = assessPrerequisitesBatch(\n options.targetPaths,\n codemodPath,\n spinners,\n );\n if (!allPassed) {\n spinners.add('prerequisite-partial-failure', {\n text: 'One or more packages failed prerequisite checks - these targets will be skipped:',\n status: 'fail',\n });\n\n for (const failedRoot of failedPackageRoots) {\n const targets = packageRootToTargets.get(failedRoot) ?? [];\n // Filter out targets that belong to failed package roots\n // eslint-disable-next-line no-param-reassign\n options.targetPaths = options.targetPaths.filter(\n (targetPath) => !targets.includes(targetPath),\n );\n spinners.add(`prerequisite-fail-${failedRoot}`, {\n text: `- \\x1b[2m\\x1b[31m${failedRoot}\\x1b[0m${targets.length > 0 && targets[0] !== failedRoot ? ` (targets: ${targets.join(', ')})` : ''}\\x1b[0m`,\n indent: 2,\n status: 'non-spinnable',\n });\n }\n }\n\n if (!options.targetPaths.length) {\n spinners.add('no-valid-targets', {\n text: `${CONSOLE_ICONS.error} No valid target paths remaining after prerequisite checks - exiting.`,\n status: 'fail',\n });\n spinners.stopAll('fail');\n spinners.checkIfActiveSpinners();\n }\n}\n","#!/usr/bin/env node\n\nimport { execSync } from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nimport { select as list } from '@inquirer/prompts';\nimport Spinnies from 'spinnies';\n\nimport { CONSOLE_ICONS } from '../constants/common';\nimport { flushTracking, track, TRACKING_EVENT, type TrackingProperties } from '../helpers/tracking';\nimport {\n assessPrerequisites,\n findPackages,\n findProjectRoot,\n getCodemodConfig,\n getOptions,\n loadTransformModules,\n logToInquirer,\n runTransformPrompts,\n validateClaudeConfig,\n} from './helpers';\nimport { handleClaudePrereqs } from './helpers/claudePrereqs';\n\nlet isDebug = false;\nconst currentFilePath = fileURLToPath(import.meta.url);\nconst currentDirPath = path.dirname(currentFilePath);\n\nconst resetReportFile = async (reportPath: string) => {\n try {\n await fs.access(reportPath);\n await fs.rm(reportPath);\n console.debug(\n `${CONSOLE_ICONS.info} Removed existing report file${isDebug ? `: ${reportPath}` : '.'}`,\n );\n } catch {\n console.debug(\n `${CONSOLE_ICONS.info} No existing report file to remove${isDebug ? `: ${reportPath}` : '.'}`,\n );\n }\n};\n\nconst summariseReportFile = async (reportPath: string) => {\n try {\n const reportContent = await fs.readFile(reportPath, 'utf8');\n const lines = reportContent.split('\\n').filter(Boolean);\n if (lines.length) {\n console.debug(\n `\\n${CONSOLE_ICONS.warning} ${lines.length} manual review${lines.length > 1 ? 's are' : ' is'} required. See ${reportPath} for details.`,\n );\n } else {\n console.debug(\n `${CONSOLE_ICONS.info} Report file exists but is empty${isDebug ? `: ${reportPath}` : '.'}`,\n );\n }\n } catch {\n console.debug(`${CONSOLE_ICONS.info} No report file generated - no manual reviews needed`);\n }\n};\n\nconst log = (label: string, value?: string): void => {\n if (typeof logToInquirer === 'function') {\n logToInquirer(label, value || '');\n } else {\n console.info(label, value || '');\n }\n};\n\nasync function runCodemod(transformsDir?: string) {\n const args = process.argv.slice(2);\n const candidate = args[0];\n isDebug = args.includes('--debug');\n const noTracking = args.includes('--no-tracking');\n const startTime = Date.now();\n\n try {\n const root = findProjectRoot();\n const packagesPromise = findPackages();\n const resolvedTransformsDir =\n transformsDir ?? path.resolve(currentDirPath, '../dist/transforms');\n\n if (isDebug) {\n console.debug(\n `${CONSOLE_ICONS.info} Resolved transforms directory: ${resolvedTransformsDir}`,\n );\n }\n\n const { transformFiles: resolvedTransformNames } =\n await loadTransformModules(resolvedTransformsDir);\n if (resolvedTransformNames.length === 0) {\n throw new Error(\n `${CONSOLE_ICONS.error} No transform scripts found${isDebug ? ` in: ${resolvedTransformsDir}` : '.'}`,\n );\n }\n\n let transformFile: string;\n\n if (candidate && resolvedTransformNames.includes(candidate)) {\n log('Select codemod to run:', candidate);\n transformFile = candidate;\n } else {\n transformFile = await list({\n message: 'Select codemod to run:',\n choices: resolvedTransformNames.map((name: string) => ({ name, value: name })),\n });\n log('Selected codemod:', transformFile);\n }\n\n const codemodPath = path.resolve(resolvedTransformsDir, transformFile, 'transformer.js');\n const codemodConfig = getCodemodConfig(codemodPath);\n if (isDebug) {\n console.debug(`${CONSOLE_ICONS.info} Resolved codemod path: ${codemodPath}`);\n }\n\n if (codemodConfig?.type === 'claude') {\n validateClaudeConfig();\n }\n\n const spinners = new Spinnies();\n spinners.add('loading-codemod', { text: `Loading codemod (${transformFile})...` });\n const packages = await packagesPromise;\n const reportPath = path.resolve(root, 'codemod-report.txt');\n spinners.succeed('loading-codemod', {\n text: `Successfully loaded codemod: \\x1b[2m${transformFile}\\x1b[0m`,\n });\n\n if (codemodConfig?.type === 'jscodeshift') {\n await resetReportFile(reportPath);\n }\n\n const promptAnswers = await runTransformPrompts(codemodPath);\n const options = await getOptions({\n packages,\n root,\n transformFiles: resolvedTransformNames,\n preselectedTransformFile: transformFile,\n transformerType: codemodConfig?.type,\n });\n\n const selectionMethod: TrackingProperties['selectionMethod'] =\n candidate && resolvedTransformNames.includes(candidate) ? 'cli' : 'interactive';\n\n const trackingProps: Omit<TrackingProperties, 'stage'> = {\n transform: transformFile,\n engine: codemodConfig?.type,\n isMonorepo: args.includes('--monorepo'),\n useGitIgnore: options.useGitIgnore,\n hasIgnorePatterns: Boolean(options.ignorePatterns),\n selectionMethod,\n targetPathCount: options.targetPaths.length,\n isDebug,\n noTracking,\n repository: path.basename(root),\n };\n\n track(TRACKING_EVENT, { ...trackingProps, stage: 'Started' });\n\n // Handle Claude transforms differently, as they work on multiple targets at once\n if (codemodConfig?.type === 'claude') {\n handleClaudePrereqs({ options, codemodPath, spinners });\n\n // Dynamically import the transformer module\n const transformerModule = (await import(codemodPath)) as {\n default: (targetPaths: string[], isDebug?: boolean) => Promise<void>;\n };\n const transformer = transformerModule.default;\n await transformer(options.targetPaths, isDebug);\n } else {\n // Button codemod doesn't use spinnies\n spinners.stopAll('succeed');\n spinners.checkIfActiveSpinners();\n\n await Promise.all(\n options.targetPaths.map(async (targetPath) => {\n console.info(\n `${CONSOLE_ICONS.focus} \\x1b[1mProcessing:\\x1b[0m \\x1b[32m${targetPath}\\x1b[0m`,\n );\n\n // Check prerequisites for this target before running\n // TODO: re-enable after testing tracking\n // const ok = assessPrerequisites(targetPath, codemodPath);\n // if (!ok) {\n // return;\n // }\n\n const answerArgs = Object.entries(promptAnswers).map(\n ([promptName, answerValue]) => `--${promptName}=${String(answerValue)}`,\n );\n\n const argsList = [\n '-t',\n codemodPath,\n targetPath,\n options.isDry ? '--dry' : '',\n options.isPrint ? '--print' : '',\n options.ignorePatterns\n ? options.ignorePatterns\n .split(',')\n .map((pattern) => `--ignore-pattern=${pattern.trim()}`)\n .join(' ')\n : '',\n options.useGitIgnore ? '--gitignore' : '',\n ...answerArgs,\n ].filter(Boolean);\n const command = `npx jscodeshift ${argsList.join(' ')}`;\n\n if (isDebug) {\n console.debug(`${CONSOLE_ICONS.info} Running: ${command}`);\n }\n\n return execSync(command, { stdio: 'inherit' });\n }),\n );\n }\n\n if (codemodConfig?.type === 'jscodeshift') {\n await summariseReportFile(reportPath);\n }\n\n track(TRACKING_EVENT, {\n ...trackingProps,\n stage: 'Finished',\n durationMs: Date.now() - startTime,\n });\n } catch (error: unknown) {\n if (error instanceof Error) {\n console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, error.message);\n } else {\n console.error(`${CONSOLE_ICONS.error} Error running ${candidate} codemod:`, String(error));\n }\n track(TRACKING_EVENT, {\n transform: candidate ?? 'unknown',\n stage: 'Failed',\n error: error instanceof Error ? error.message : String(error),\n durationMs: Date.now() - startTime,\n });\n\n await flushTracking();\n\n if (process.env.NODE_ENV !== 'test') {\n process.exit(1);\n }\n }\n\n await flushTracking();\n}\n\nexport { runCodemod };\n","#!/usr/bin/env node\nimport { runCodemod } from './controller';\n\nvoid runCodemod();\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@wise/wds-codemods",
3
- "version": "1.2.1-experimental-18899d8",
3
+ "version": "1.2.1-experimental-153ecf4",
4
4
  "license": "UNLICENSED",
5
5
  "author": "Wise Payments Ltd.",
6
6
  "repository": {