UNPKG

astx

Version:

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

287 lines (252 loc) 29.4 kB
import { mapValues } from 'lodash-es' import compileMatcher, { mergeCaptures } from './compileMatcher/index.mjs' import ensureArray from './util/ensureArray.mjs' import forEachNode from './util/forEachNode.mjs' export function convertWithCaptures(matches) { return mergeCaptures( ...ensureArray(matches).map( ({ pathCaptures, arrayPathCaptures, stringCaptures }) => ({ captures: pathCaptures, arrayCaptures: arrayPathCaptures, stringCaptures, }) ) ) } export function createMatch(paths, result) { var _paths$ if (!result) { throw new Error('result must be defined') } const { captures, arrayCaptures, stringCaptures } = result const match = Array.isArray(paths) ? { type: 'nodes', node: (_paths$ = paths[0]) === null || _paths$ === void 0 ? void 0 : _paths$.node, path: paths[0], nodes: paths.map((p) => p.node), paths, } : { type: 'node', node: paths.node, path: paths, nodes: [paths.node], paths: [paths], } if (captures) { match.pathCaptures = captures match.captures = mapValues(captures, (path) => path.node) } if (arrayCaptures) { match.arrayPathCaptures = arrayCaptures match.arrayCaptures = mapValues(arrayCaptures, (paths) => paths.map((path) => path.node) ) } if (stringCaptures) match.stringCaptures = stringCaptures return match } export default function find(paths, pattern, options) { const t = options.backend.t const n = t.namedTypes if (Array.isArray(pattern) && pattern.length === 1) pattern = pattern[0] if (Array.isArray(pattern)) { if (!n.Statement.check(pattern[0].value)) { throw new Error(`pattern array must be an array of statements`) } return findStatements(paths, pattern, options) } const matcher = compileMatcher(pattern, options) const matches = [] const nodeTypes = ensureArray(matcher.nodeType || 'Node') forEachNode(t, ensureArray(paths), nodeTypes, (path) => { var _options$matchSoFar const result = matcher.match( path, (_options$matchSoFar = options === null || options === void 0 ? void 0 : options.matchSoFar) !== null && _options$matchSoFar !== void 0 ? _options$matchSoFar : null ) if (result) matches.push(createMatch(path, result)) }) return matches } function findStatements(paths, pattern, options) { var _options$matchSoFar2 const t = options.backend.t const matchers = pattern.map((queryElem) => compileMatcher(queryElem, options) ) const firstNonArrayCaptureIndex = matchers.findIndex( (m) => !m.arrayPlaceholder ) if (firstNonArrayCaptureIndex < 0) { throw new Error( `pattern would match every single array of statements, this is unsupported` ) } function remainingElements(matcherIndex) { let count = 0 for (let i = matcherIndex; i < matchers.length; i++) { if (!matchers[i].arrayPlaceholder) count++ } return count } function matchElem(paths, sliceStart, arrayIndex, matcherIndex, matchSoFar) { if (arrayIndex === paths.length || matcherIndex === matchers.length) { return remainingElements(matcherIndex) === 0 ? [matchSoFar || {}, [arrayIndex, arrayIndex]] : null } const matcher = matchers[matcherIndex] const { arrayPlaceholder } = matcher if (arrayPlaceholder) { if (matcherIndex === matchers.length - 1) { return [ mergeCaptures(matchSoFar, { arrayCaptures: { [arrayPlaceholder]: paths.slice(sliceStart), }, }), [sliceStart, paths.length], ] } return matchElem( paths, sliceStart, arrayIndex, matcherIndex + 1, matchSoFar ) } else { var _matchers const origMatchSoFar = matchSoFar const prevArrayPlaceholder = (_matchers = matchers[matcherIndex - 1]) === null || _matchers === void 0 ? void 0 : _matchers.arrayPlaceholder const end = prevArrayPlaceholder && matcherIndex !== firstNonArrayCaptureIndex ? paths.length - remainingElements(matcherIndex + 1) : arrayIndex + 1 for (let i = arrayIndex; i < end; i++) { matchSoFar = matcher.match(paths[i], origMatchSoFar) if (!matchSoFar) continue if (prevArrayPlaceholder) { matchSoFar = mergeCaptures(matchSoFar, { arrayCaptures: { [prevArrayPlaceholder]: paths.slice(sliceStart, i), }, }) } const restMatch = matchElem( paths, i + 1, i + 1, matcherIndex + 1, matchSoFar ) if (restMatch) return [restMatch[0], [i, restMatch[1][1]]] } } return null } const blocks = [] forEachNode(t, ensureArray(paths), ['Block'], (path) => { blocks.push(path) }) blocks.reverse() const matches = [] const initialMatch = (_options$matchSoFar2 = options === null || options === void 0 ? void 0 : options.matchSoFar) !== null && _options$matchSoFar2 !== void 0 ? _options$matchSoFar2 : null for (const block of blocks) { const body = block.get('body').filter(() => true) const end = body.length - remainingElements(firstNonArrayCaptureIndex + 1) let sliceStart = 0 for (let arrayIndex = 0; arrayIndex < end; arrayIndex++) { const match = matchElem( body, sliceStart, arrayIndex, firstNonArrayCaptureIndex, initialMatch ) if (match) { var _result2, _result3, _result4 let result = match[0] const start = firstNonArrayCaptureIndex > 0 ? 0 : match[1][0] const [, end] = match[1] // make sure all * captures are present in results // (if there are more than one adjacent *, all captured paths will be in the // last one and the rest will be empty) for (const matcher of matchers) { var _result, _result$arrayCaptures const { arrayPlaceholder } = matcher if (!arrayPlaceholder) continue if ( !( (_result = result) !== null && _result !== void 0 && (_result$arrayCaptures = _result.arrayCaptures) !== null && _result$arrayCaptures !== void 0 && _result$arrayCaptures[arrayPlaceholder] ) ) result = mergeCaptures(result, { arrayCaptures: { [arrayPlaceholder]: [], }, }) } const paths = body.slice(start, end) const finalMatch = { type: 'nodes', path: paths[0], node: paths[0].node, paths, nodes: paths.map((p) => p.node), } if ( (_result2 = result) !== null && _result2 !== void 0 && _result2.captures ) { finalMatch.pathCaptures = result.captures finalMatch.captures = mapValues(result.captures, (p) => p.node) } if ( (_result3 = result) !== null && _result3 !== void 0 && _result3.arrayCaptures ) { finalMatch.arrayPathCaptures = result.arrayCaptures finalMatch.arrayCaptures = mapValues(result.arrayCaptures, (paths) => paths.map((p) => p.node) ) } if ( (_result4 = result) !== null && _result4 !== void 0 && _result4.stringCaptures ) { finalMatch.stringCaptures = result.stringCaptures } matches.push(finalMatch) // prevent overlapping matches sliceStart = end arrayIndex = end - 1 } } } return matches } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["mapValues","compileMatcher","mergeCaptures","ensureArray","forEachNode","convertWithCaptures","matches","map","pathCaptures","arrayPathCaptures","stringCaptures","captures","arrayCaptures","createMatch","paths","result","Error","match","Array","isArray","type","node","path","nodes","p","find","pattern","options","t","backend","n","namedTypes","length","Statement","check","value","findStatements","matcher","nodeTypes","nodeType","matchSoFar","push","matchers","queryElem","firstNonArrayCaptureIndex","findIndex","m","arrayPlaceholder","remainingElements","matcherIndex","count","i","matchElem","sliceStart","arrayIndex","slice","origMatchSoFar","prevArrayPlaceholder","end","restMatch","blocks","reverse","initialMatch","block","body","get","filter","start","finalMatch"],"sources":["../src/find.ts"],"sourcesContent":["import { NodeType, NodePath, Node, Statement, Block } from './types'\nimport { mapValues } from 'lodash'\nimport compileMatcher, {\n  CompiledMatcher,\n  MatchResult,\n  mergeCaptures,\n} from './compileMatcher'\nimport { Backend } from './backend/Backend'\nimport ensureArray from './util/ensureArray'\nimport forEachNode from './util/forEachNode'\n\nexport type Match = {\n  type: 'node' | 'nodes'\n  path: NodePath\n  node: Node\n  paths: NodePath[]\n  nodes: Node[]\n  pathCaptures?: Record<string, NodePath>\n  captures?: Record<string, Node>\n  arrayPathCaptures?: Record<string, NodePath[]>\n  arrayCaptures?: Record<string, Node[]>\n  stringCaptures?: Record<string, string>\n}\n\nexport type FindOptions = {\n  backend: Backend\n  where?: { [captureName: string]: (path: NodePath) => boolean }\n  matchSoFar?: MatchResult\n}\n\nexport function convertWithCaptures(matches: Match | Match[]): MatchResult {\n  return mergeCaptures(\n    ...ensureArray(matches).map(\n      ({ pathCaptures, arrayPathCaptures, stringCaptures }): MatchResult => ({\n        captures: pathCaptures,\n        arrayCaptures: arrayPathCaptures,\n        stringCaptures,\n      })\n    )\n  )\n}\n\nexport function createMatch(\n  paths: NodePath | NodePath[],\n  result: MatchResult\n): Match {\n  if (!result) {\n    throw new Error('result must be defined')\n  }\n  const { captures, arrayCaptures, stringCaptures } = result\n  const match: Match = Array.isArray(paths)\n    ? {\n        type: 'nodes',\n        node: paths[0]?.node,\n        path: paths[0],\n        nodes: paths.map((p) => p.node),\n        paths,\n      }\n    : {\n        type: 'node',\n        node: paths.node,\n        path: paths,\n        nodes: [paths.node],\n        paths: [paths],\n      }\n  if (captures) {\n    match.pathCaptures = captures\n    match.captures = mapValues(captures, (path: NodePath) => path.node)\n  }\n  if (arrayCaptures) {\n    match.arrayPathCaptures = arrayCaptures\n    match.arrayCaptures = mapValues(arrayCaptures, (paths: NodePath[]) =>\n      paths.map((path) => path.node)\n    )\n  }\n  if (stringCaptures) match.stringCaptures = stringCaptures\n  return match\n}\n\nexport default function find(\n  paths: NodePath | readonly NodePath[],\n  pattern: NodePath | readonly NodePath[],\n  options: FindOptions\n): Match[] {\n  const t = options.backend.t\n  const n = t.namedTypes\n  if (Array.isArray(pattern) && pattern.length === 1) pattern = pattern[0]\n  if (Array.isArray(pattern)) {\n    if (!n.Statement.check(pattern[0].value)) {\n      throw new Error(`pattern array must be an array of statements`)\n    }\n    return findStatements(\n      paths,\n      pattern as NodePath<Statement, Statement>[],\n      options\n    )\n  }\n\n  const matcher = compileMatcher(pattern as NodePath, options)\n\n  const matches: Array<Match> = []\n\n  const nodeTypes: readonly NodeType[] = ensureArray(matcher.nodeType || 'Node')\n\n  forEachNode(t, ensureArray(paths), nodeTypes, (path: NodePath) => {\n    const result = matcher.match(path, options?.matchSoFar ?? null)\n    if (result) matches.push(createMatch(path, result))\n  })\n\n  return matches\n}\n\nfunction findStatements(\n  paths: NodePath | readonly NodePath[],\n  pattern: readonly NodePath<Statement, Statement>[],\n  options: FindOptions\n): Match[] {\n  const t = options.backend.t\n\n  const matchers: CompiledMatcher[] = pattern.map((queryElem) =>\n    compileMatcher(queryElem, options)\n  )\n\n  const firstNonArrayCaptureIndex = matchers.findIndex(\n    (m) => !m.arrayPlaceholder\n  )\n\n  if (firstNonArrayCaptureIndex < 0) {\n    throw new Error(\n      `pattern would match every single array of statements, this is unsupported`\n    )\n  }\n\n  function remainingElements(matcherIndex: number): number {\n    let count = 0\n    for (let i = matcherIndex; i < matchers.length; i++) {\n      if (!matchers[i].arrayPlaceholder) count++\n    }\n    return count\n  }\n\n  function matchElem(\n    paths: NodePath<Statement>[],\n    sliceStart: number,\n    arrayIndex: number,\n    matcherIndex: number,\n    matchSoFar: MatchResult\n  ): [MatchResult, [number, number]] | null {\n    if (arrayIndex === paths.length || matcherIndex === matchers.length) {\n      return remainingElements(matcherIndex) === 0\n        ? [matchSoFar || {}, [arrayIndex, arrayIndex]]\n        : null\n    }\n\n    const matcher = matchers[matcherIndex]\n    const { arrayPlaceholder } = matcher\n    if (arrayPlaceholder) {\n      if (matcherIndex === matchers.length - 1) {\n        return [\n          mergeCaptures(matchSoFar, {\n            arrayCaptures: {\n              [arrayPlaceholder]: paths.slice(sliceStart),\n            },\n          }),\n          [sliceStart, paths.length],\n        ]\n      }\n      return matchElem(\n        paths,\n        sliceStart,\n        arrayIndex,\n        matcherIndex + 1,\n        matchSoFar\n      )\n    } else {\n      const origMatchSoFar = matchSoFar\n      const prevArrayPlaceholder = matchers[matcherIndex - 1]?.arrayPlaceholder\n      const end =\n        prevArrayPlaceholder && matcherIndex !== firstNonArrayCaptureIndex\n          ? paths.length - remainingElements(matcherIndex + 1)\n          : arrayIndex + 1\n      for (let i = arrayIndex; i < end; i++) {\n        matchSoFar = matcher.match(paths[i], origMatchSoFar)\n        if (!matchSoFar) continue\n        if (prevArrayPlaceholder) {\n          matchSoFar = mergeCaptures(matchSoFar, {\n            arrayCaptures: {\n              [prevArrayPlaceholder]: paths.slice(sliceStart, i),\n            },\n          })\n        }\n        const restMatch = matchElem(\n          paths,\n          i + 1,\n          i + 1,\n          matcherIndex + 1,\n          matchSoFar\n        )\n        if (restMatch) return [restMatch[0], [i, restMatch[1][1]]]\n      }\n    }\n    return null\n  }\n\n  const blocks: NodePath<Block>[] = []\n\n  forEachNode(t, ensureArray(paths), ['Block'], (path: NodePath) => {\n    blocks.push(path as NodePath<Block>)\n  })\n  blocks.reverse()\n\n  const matches: Match[] = []\n\n  const initialMatch = options?.matchSoFar ?? null\n\n  for (const block of blocks) {\n    const body: NodePath<Statement>[] = block.get('body').filter(() => true)\n    const end = body.length - remainingElements(firstNonArrayCaptureIndex + 1)\n    let sliceStart = 0\n    for (let arrayIndex = 0; arrayIndex < end; arrayIndex++) {\n      const match = matchElem(\n        body,\n        sliceStart,\n        arrayIndex,\n        firstNonArrayCaptureIndex,\n        initialMatch\n      )\n      if (match) {\n        let result = match[0]\n        const start = firstNonArrayCaptureIndex > 0 ? 0 : match[1][0]\n        const [, end] = match[1]\n\n        // make sure all * captures are present in results\n        // (if there are more than one adjacent *, all captured paths will be in the\n        // last one and the rest will be empty)\n        for (const matcher of matchers) {\n          const { arrayPlaceholder } = matcher\n          if (!arrayPlaceholder) continue\n          if (!result?.arrayCaptures?.[arrayPlaceholder])\n            result = mergeCaptures(result, {\n              arrayCaptures: { [arrayPlaceholder]: [] },\n            })\n        }\n\n        const paths = body.slice(start, end)\n\n        const finalMatch: Match = {\n          type: 'nodes',\n          path: paths[0],\n          node: paths[0].node,\n          paths,\n          nodes: paths.map((p) => p.node),\n        }\n        if (result?.captures) {\n          finalMatch.pathCaptures = result.captures\n          finalMatch.captures = mapValues(result.captures, (p) => p.node)\n        }\n        if (result?.arrayCaptures) {\n          finalMatch.arrayPathCaptures = result.arrayCaptures\n          finalMatch.arrayCaptures = mapValues(result.arrayCaptures, (paths) =>\n            paths.map((p) => p.node)\n          )\n        }\n        if (result?.stringCaptures) {\n          finalMatch.stringCaptures = result.stringCaptures\n        }\n\n        matches.push(finalMatch)\n\n        // prevent overlapping matches\n        sliceStart = end\n        arrayIndex = end - 1\n      }\n    }\n  }\n  return matches\n}\n"],"mappings":"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"}