astx
Version:
super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring
164 lines (152 loc) • 16.7 kB
JavaScript
import compileMatcher from './index.mjs'
import indentDebug from './indentDebug.mjs'
import areFieldValuesEqual from '../util/areFieldValuesEqual.mjs'
const equivalenceClassesArray = [
{
nodeTypes: new Set(['ClassDeclaration', 'ClassExpression']),
},
{
nodeTypes: new Set([
'FunctionDeclaration',
'FunctionExpression',
'ArrowFunctionExpression',
'ObjectMethod',
'ClassMethod',
'ClassPrivateMethod',
]),
baseType: 'Function',
},
{
nodeTypes: new Set(['Identifier', 'JSXIdentifier']),
},
]
const equivalenceClasses = {}
for (const klass of equivalenceClassesArray) {
for (const type of klass.nodeTypes) equivalenceClasses[type] = klass
}
const normalizeNull = (value) => {
return value === undefined ? null : value
}
export default function compileGenericNodeMatcher(
path,
compileOptions,
options
) {
const {
backend: { t },
} = compileOptions
const pattern = path.node
const { baseType, nodeTypes } = equivalenceClasses[pattern.type] || {}
const nodeType =
baseType || (nodeTypes ? [...nodeTypes] : null) || pattern.type
const isType = baseType
? (node) => {
var _baseType
return (_baseType = t.namedTypes[baseType]) === null ||
_baseType === void 0
? void 0
: _baseType.check(node)
}
: null
const isCorrectType = isType
? (node) => isType(node)
: nodeTypes
? (node) =>
nodeTypes.has(node === null || node === void 0 ? void 0 : node.type)
: (node) =>
(node === null || node === void 0 ? void 0 : node.type) === pattern.type
const { debug } = compileOptions
const keyMatchers = Object.fromEntries(
t.getFieldNames(pattern).map((key) => {
var _options$keyMatchers
const custom =
options === null || options === void 0
? void 0
: (_options$keyMatchers = options.keyMatchers) === null ||
_options$keyMatchers === void 0
? void 0
: _options$keyMatchers[key]
if (custom) return [key, custom]
const value = normalizeNull(t.getFieldValue(pattern, key))
const fieldPath = path.get(key)
if (Array.isArray(value) || fieldPath.node === value) {
return [
key,
compileMatcher(fieldPath, {
...compileOptions,
debug: indentDebug(debug, 2),
}),
]
} else if (value instanceof Object) {
return [
key,
{
pattern: fieldPath,
match: (path, matchSoFar) => {
const nodeValue = normalizeNull(path.value)
if (areFieldValuesEqual(t, value, nodeValue)) {
debug(' %s === %s', value, nodeValue)
return matchSoFar || {}
} else {
debug(' %s !== %s', value, nodeValue)
return null
}
},
},
]
} else {
return [
key,
{
pattern: fieldPath,
match: (path, matchSoFar) => {
const nodeValue = normalizeNull(path.value)
if (
value === nodeValue ||
(value === null && nodeValue === false) ||
(value === false && nodeValue === null)
) {
debug(' %s === %s', value, nodeValue)
return matchSoFar || {}
} else {
debug(' %s !== %s', value, nodeValue)
return null
}
},
},
]
}
})
)
return {
pattern: path,
match: (path, matchSoFar) => {
debug('%s (generic)', pattern.type)
if (Array.isArray(path.value)) return null
if (
isCorrectType(path === null || path === void 0 ? void 0 : path.value)
) {
for (const key in keyMatchers) {
debug(' .%s', key)
const matcher = keyMatchers[key]
matchSoFar = matcher.match(path.get(key), matchSoFar)
if (!matchSoFar) return null
}
return matchSoFar || {}
} else {
var _path$value
debug(
' path?.value?.type (%s) is not compatible with pattern.type (%s)',
path === null || path === void 0
? void 0
: (_path$value = path.value) === null || _path$value === void 0
? void 0
: _path$value.type,
pattern.type
)
return null
}
},
nodeType,
}
} //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["compileMatcher","indentDebug","areFieldValuesEqual","equivalenceClassesArray","nodeTypes","Set","baseType","equivalenceClasses","klass","type","normalizeNull","value","undefined","compileGenericNodeMatcher","path","compileOptions","options","backend","t","pattern","node","nodeType","isType","namedTypes","check","isCorrectType","has","debug","keyMatchers","Object","fromEntries","getFieldNames","map","key","custom","getFieldValue","fieldPath","get","Array","isArray","match","matchSoFar","nodeValue","matcher"],"sources":["../../src/compileMatcher/GenericNodeMatcher.ts"],"sourcesContent":["import { Node, NodePath, NodeType } from '../types'\nimport compileMatcher, {\n  CompiledMatcher,\n  CompileOptions,\n  MatchResult,\n} from './index'\nimport indentDebug from './indentDebug'\nimport areFieldValuesEqual from '../util/areFieldValuesEqual'\n\nconst equivalenceClassesArray: {\n  nodeTypes: Set<NodeType>\n  baseType?: NodeType\n}[] = [\n  { nodeTypes: new Set(['ClassDeclaration', 'ClassExpression']) },\n  {\n    nodeTypes: new Set([\n      'FunctionDeclaration',\n      'FunctionExpression',\n      'ArrowFunctionExpression',\n      'ObjectMethod',\n      'ClassMethod',\n      'ClassPrivateMethod',\n    ]),\n    baseType: 'Function',\n  },\n  {\n    nodeTypes: new Set(['Identifier', 'JSXIdentifier']),\n  },\n]\n\nconst equivalenceClasses: Partial<\n  Record<NodeType, { nodeTypes: Set<NodeType>; baseType?: NodeType }>\n> = {}\nfor (const klass of equivalenceClassesArray) {\n  for (const type of klass.nodeTypes) equivalenceClasses[type] = klass\n}\n\nconst normalizeNull = <T>(value: T | null | undefined): T | null => {\n  return value === undefined ? null : value\n}\n\nexport default function compileGenericNodeMatcher(\n  path: NodePath<Node, Node>,\n  compileOptions: CompileOptions,\n  options?: {\n    keyMatchers?: Record<string, CompiledMatcher>\n  }\n): CompiledMatcher {\n  const {\n    backend: { t },\n  } = compileOptions\n\n  const pattern: Node = path.node\n\n  const { baseType, nodeTypes } =\n    equivalenceClasses[pattern.type as NodeType] || {}\n\n  const nodeType =\n    baseType ||\n    (nodeTypes ? [...nodeTypes] : null) ||\n    (pattern.type as NodeType)\n  const isType = baseType\n    ? (node: any) => (t.namedTypes as any)[baseType]?.check(node)\n    : null\n  const isCorrectType = isType\n    ? (node: Node) => isType(node)\n    : nodeTypes\n    ? (node: Node) => nodeTypes.has(node?.type as NodeType)\n    : (node: Node) => node?.type === pattern.type\n\n  const { debug } = compileOptions\n\n  const keyMatchers: Record<string, CompiledMatcher> = Object.fromEntries(\n    t.getFieldNames(pattern).map((key: string): [string, CompiledMatcher] => {\n      const custom = options?.keyMatchers?.[key]\n\n      if (custom) return [key, custom]\n\n      const value = normalizeNull(t.getFieldValue(pattern, key))\n      const fieldPath = path.get(key)\n\n      if (Array.isArray(value) || fieldPath.node === value) {\n        return [\n          key,\n          compileMatcher(fieldPath, {\n            ...compileOptions,\n            debug: indentDebug(debug, 2),\n          }),\n        ]\n      } else if (value instanceof Object) {\n        return [\n          key,\n          {\n            pattern: fieldPath,\n            match: (path: NodePath, matchSoFar: MatchResult): MatchResult => {\n              const nodeValue = normalizeNull(path.value)\n\n              if (areFieldValuesEqual(t, value, nodeValue)) {\n                debug('    %s === %s', value, nodeValue)\n                return matchSoFar || {}\n              } else {\n                debug('    %s !== %s', value, nodeValue)\n                return null\n              }\n            },\n          },\n        ]\n      } else {\n        return [\n          key,\n          {\n            pattern: fieldPath,\n            match: (path: NodePath, matchSoFar: MatchResult): MatchResult => {\n              const nodeValue = normalizeNull(path.value)\n\n              if (\n                value === nodeValue ||\n                (value === null && nodeValue === false) ||\n                (value === false && nodeValue === null)\n              ) {\n                debug('    %s === %s', value, nodeValue)\n                return matchSoFar || {}\n              } else {\n                debug('    %s !== %s', value, nodeValue)\n                return null\n              }\n            },\n          },\n        ]\n      }\n    })\n  )\n\n  return {\n    pattern: path,\n    match: (path: NodePath, matchSoFar: MatchResult): MatchResult => {\n      debug('%s (generic)', pattern.type)\n\n      if (Array.isArray(path.value)) return null\n\n      if (isCorrectType(path?.value)) {\n        for (const key in keyMatchers) {\n          debug('  .%s', key)\n          const matcher = keyMatchers[key]\n          matchSoFar = matcher.match(path.get(key), matchSoFar)\n          if (!matchSoFar) return null\n        }\n\n        return matchSoFar || {}\n      } else {\n        debug(\n          '  path?.value?.type (%s) is not compatible with pattern.type (%s)',\n          path?.value?.type,\n          pattern.type\n        )\n\n        return null\n      }\n    },\n\n    nodeType,\n  }\n}\n"],"mappings":"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"}