astx
Version:
super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring
287 lines (252 loc) • 29.4 kB
JavaScript
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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"}