UNPKG

astx

Version:

super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring

197 lines (187 loc) 20.2 kB
import defaultFs from 'fs-extra' import Path from 'path' import { memoize } from 'lodash-es' import { promisify } from 'util' import _resolve from 'resolve' import omitBlankLineChanges from '../util/omitBlankLineChanges.mjs' import CodeFrameError from '../util/CodeFrameError.mjs' import chooseGetBackend from '../chooseGetBackend.mjs' import { astxCosmiconfig } from './astxCosmiconfig.mjs' import Astx from '../Astx.mjs' import './registerTsNode.mjs' const resolve = promisify(_resolve) const getPrettier = memoize(async (path) => { try { // eslint-disable-next-line @typescript-eslint/no-var-requires let prettier = await import( /* webpackIgnore: true */ await resolve('prettier', { basedir: path, }) ) if (prettier.default instanceof Object) { prettier = prettier.default } if ( typeof prettier.format === 'function' && typeof prettier.resolveConfig === 'function' ) { return prettier } } catch (error) { // ignore } return null }) export default async function runTransformOnFile({ transform: _transform, transformFile, config: configOverrides, file, source, signal, forWorker, fs = defaultFs, }) { var _await$astxCosmiconfi const transform = transformFile ? await import(transformFile) : _transform const baseConfig = (_await$astxCosmiconfi = await astxCosmiconfig.search( Path.dirname(file) )) === null || _await$astxCosmiconfi === void 0 ? void 0 : _await$astxCosmiconfi.config const config = { ...baseConfig, ...configOverrides, parserOptions: (baseConfig !== null && baseConfig !== void 0 && baseConfig.parserOptions) || (configOverrides !== null && configOverrides !== void 0 && configOverrides.parserOptions) ? { ...(baseConfig === null || baseConfig === void 0 ? void 0 : baseConfig.parserOptions), ...(configOverrides === null || configOverrides === void 0 ? void 0 : configOverrides.parserOptions), } : undefined, } if (signal !== null && signal !== void 0 && signal.aborted) throw new Error('aborted') const { parser, parserOptions } = config const backend = await chooseGetBackend(parser)(file, parserOptions) if (signal !== null && signal !== void 0 && signal.aborted) throw new Error('aborted') try { if (!source) source = await fs.readFile(file, 'utf8') if (signal !== null && signal !== void 0 && signal.aborted) throw new Error('aborted') let transformed const reports = [] let matches let transformFn = transform.astx const { find, replace } = transform if (typeof transformFn !== 'function' && find) { transformFn = ({ astx }) => { const result = astx.find(find, { where: transform.where, }) if (replace) result.replace(replace) matches = result.matches if (!result.size) return null } } if (typeof transformFn === 'function') { let ast, root try { ast = backend.parse(source) root = new backend.t.NodePath(ast) } catch (error) { if (error instanceof Error) { CodeFrameError.rethrow(error, { filename: file, source, }) } throw error } const options = { source, file, root, t: backend.t, report: (msg) => { var _transform$onReport if (msg instanceof Astx && !msg.size) return if (!forWorker) (_transform$onReport = transform.onReport) === null || _transform$onReport === void 0 ? void 0 : _transform$onReport.call(transform, { file, report: msg, }) reports.push(msg) }, ...backend.template, astx: new Astx(backend, [root]), } const [_result, prettier] = await Promise.all([ transformFn(options), (config === null || config === void 0 ? void 0 : config.prettier) !== false ? getPrettier(Path.dirname(file)) : null, ]) if (signal !== null && signal !== void 0 && signal.aborted) throw new Error('aborted') if (transform.astx || transform.replace) { transformed = _result if (transformed === undefined) { transformed = backend.generate(ast).code } if (transformed === null) transformed = undefined if ( prettier && typeof transformed === 'string' && transformed !== source ) { const prettierConfig = (await prettier.resolveConfig(file)) || {} prettierConfig.filepath = file if (/\.tsx?$/.test(file)) prettierConfig.parser = 'typescript' transformed = prettier.format(transformed, prettierConfig) } if (transformed != null) { transformed = omitBlankLineChanges(source, transformed) } } } else { return { file, error: new Error( 'transform file must export either astx or find/replace' ), backend, } } return { file, source, transformed, reports, matches, backend, } } catch (error) { return { file, error: error instanceof Error ? error : new Error(String(error)), backend, } } } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["defaultFs","Path","memoize","promisify","_resolve","omitBlankLineChanges","CodeFrameError","chooseGetBackend","astxCosmiconfig","Astx","resolve","getPrettier","path","prettier","basedir","default","Object","format","resolveConfig","error","runTransformOnFile","transform","_transform","transformFile","config","configOverrides","file","source","signal","forWorker","fs","baseConfig","search","dirname","parserOptions","undefined","aborted","Error","parser","backend","readFile","transformed","reports","matches","transformFn","astx","find","replace","result","where","size","ast","root","parse","t","NodePath","rethrow","filename","options","report","msg","onReport","push","template","_result","Promise","all","generate","code","prettierConfig","filepath","test","String"],"sources":["../../src/node/runTransformOnFile.ts"],"sourcesContent":["import defaultFs from 'fs-extra'\nimport Path from 'path'\nimport { memoize } from 'lodash'\nimport { promisify } from 'util'\nimport _resolve from 'resolve'\nimport { Match } from '../find'\nimport omitBlankLineChanges from '../util/omitBlankLineChanges'\nimport CodeFrameError from '../util/CodeFrameError'\nimport { AstxConfig } from '../AstxConfig'\nimport chooseGetBackend from '../chooseGetBackend'\nimport { astxCosmiconfig } from './astxCosmiconfig'\nimport Astx, { Transform, TransformOptions, TransformResult } from '../Astx'\nimport { Node } from '../types'\nimport './registerTsNode'\nconst resolve = promisify(_resolve) as any\n\nconst getPrettier = memoize(async (path: string): Promise<any> => {\n  try {\n    // eslint-disable-next-line @typescript-eslint/no-var-requires\n    let prettier = await import(\n      /* webpackIgnore: true */\n      await resolve('prettier', {\n        basedir: path,\n      })\n    )\n    if (prettier.default instanceof Object) {\n      prettier = prettier.default\n    }\n    if (\n      typeof prettier.format === 'function' &&\n      typeof prettier.resolveConfig === 'function'\n    ) {\n      return prettier\n    }\n  } catch (error) {\n    // ignore\n  }\n  return null\n})\n\nexport interface Fs {\n  readFile(file: string, encoding: string): Promise<string>\n}\n\nexport type RunTransformOnFileOptions = {\n  file: string\n  source?: string\n  transform?: Transform\n  transformFile?: string\n  config?: Partial<AstxConfig>\n  signal?: AbortSignal\n  forWorker?: boolean\n  fs?: Fs\n}\n\nexport default async function runTransformOnFile({\n  transform: _transform,\n  transformFile,\n  config: configOverrides,\n  file,\n  source,\n  signal,\n  forWorker,\n  fs = defaultFs,\n}: RunTransformOnFileOptions): Promise<TransformResult> {\n  const transform: Transform = transformFile\n    ? await import(transformFile)\n    : _transform\n\n  const baseConfig = (await astxCosmiconfig.search(Path.dirname(file)))\n    ?.config as AstxConfig | undefined\n\n  const config: AstxConfig = {\n    ...baseConfig,\n    ...configOverrides,\n    parserOptions:\n      baseConfig?.parserOptions || configOverrides?.parserOptions\n        ? { ...baseConfig?.parserOptions, ...configOverrides?.parserOptions }\n        : undefined,\n  }\n\n  if (signal?.aborted) throw new Error('aborted')\n\n  const { parser, parserOptions } = config\n\n  const backend = await chooseGetBackend(parser)(file, parserOptions)\n  if (signal?.aborted) throw new Error('aborted')\n\n  try {\n    if (!source) source = await fs.readFile(file, 'utf8')\n    if (signal?.aborted) throw new Error('aborted')\n\n    let transformed\n    const reports: unknown[] = []\n\n    let matches: readonly Match[] | undefined\n\n    let transformFn = transform.astx\n\n    const { find, replace } = transform\n    if (typeof transformFn !== 'function' && find) {\n      transformFn = ({ astx }: TransformOptions): any => {\n        const result = astx.find(find as string | Node | Node[], {\n          where: transform.where,\n        })\n        if (replace) result.replace(replace)\n        matches = result.matches\n        if (!result.size) return null\n      }\n    }\n    if (typeof transformFn === 'function') {\n      let ast, root\n      try {\n        ast = backend.parse(source)\n        root = new backend.t.NodePath(ast)\n      } catch (error) {\n        if (error instanceof Error) {\n          CodeFrameError.rethrow(error, { filename: file, source })\n        }\n        throw error\n      }\n      const options = {\n        source,\n        file,\n        root,\n        t: backend.t,\n        report: (msg: unknown) => {\n          if (msg instanceof Astx && !msg.size) return\n          if (!forWorker) transform.onReport?.({ file, report: msg })\n          reports.push(msg)\n        },\n        ...backend.template,\n        astx: new Astx(backend, [root]),\n      }\n      const [_result, prettier] = await Promise.all([\n        transformFn(options),\n        config?.prettier !== false ? getPrettier(Path.dirname(file)) : null,\n      ])\n      if (signal?.aborted) throw new Error('aborted')\n      if (transform.astx || transform.replace) {\n        transformed = _result\n        if (transformed === undefined) {\n          transformed = backend.generate(ast).code\n        }\n        if (transformed === null) transformed = undefined\n        if (\n          prettier &&\n          typeof transformed === 'string' &&\n          transformed !== source\n        ) {\n          const prettierConfig = (await prettier.resolveConfig(file)) || {}\n          prettierConfig.filepath = file\n          if (/\\.tsx?$/.test(file)) prettierConfig.parser = 'typescript'\n          transformed = prettier.format(transformed, prettierConfig)\n        }\n        if (transformed != null) {\n          transformed = omitBlankLineChanges(source, transformed)\n        }\n      }\n    } else {\n      return {\n        file,\n        error: new Error(\n          'transform file must export either astx or find/replace'\n        ),\n        backend,\n      }\n    }\n    return {\n      file,\n      source,\n      transformed,\n      reports,\n      matches,\n      backend,\n    }\n  } catch (error) {\n    return {\n      file,\n      error: error instanceof Error ? error : new Error(String(error)),\n      backend,\n    }\n  }\n}\n"],"mappings":"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"}