zimmerframe
A tool for walking.
Specifically, it's a tool for walking an abstract syntax tree (AST), where every node is an object with a type: string
. This includes ESTree nodes, such as you might generate with Acorn or Meriyah, but also includes things like CSSTree or an arbitrary AST of your own devising.
Usage
import { walk } from 'zimmerframe';
import { parse } from 'acorn';
import { Node } from 'estree';
const program = parse(`
let message = 'hello';
console.log(message);
if (true) {
let answer = 42;
console.log(answer);
}
`);
const state = {
declarations: [],
depth: 0
};
const transformed = walk(program as Node, state, {
_(node, { state }) {
},
VariableDeclarator(node, { state }) {
if (node.id.type === 'Identifier') {
state.declarations.push({
depth: state.depth,
name: node.id.name
});
}
},
BlockStatement(node, { state, next, skip, stop }) {
console.log('entering BlockStatement');
next({ ...state, depth: state.depth + 1 });
console.log('leaving BlockStatement');
},
Literal(node) {
if (node.value === 'hello') {
return {
...node,
value: 'goodbye'
};
}
},
IfStatement(node, { visit }) {
if (node.test.type === 'Literal' && node.test.value === true) {
return visit(node.consequent);
}
}
});
The transformed
AST would look like this:
let message = 'goodbye';
console.log(message);
{
let answer = 42;
console.log(answer);
}
Types
The type of node
in each visitor is inferred from the visitor's name. For example:
walk(ast as estree.Node, state, {
ArrowFunctionExpression(node) {
}
});
For this to work, the first argument should be casted to an union of all the types you plan to visit.
You can import types from 'zimmerframe':
import {
walk,
type Visitor,
type Visitors,
type Context
} from 'zimmerframe';
import type { Node } from 'estree';
interface State {...}
const node: Node = {...};
const state: State = {...};
const visitors: Visitors<Node, State> = {...}
walk(node, state, visitors);
Context
Each visitor receives a second argument, context
, which is an object with the following properties and methods:
path: Node[]
— an array of parent nodes. For example, to get the root node you would do path.at(0)
; to get the current node's immediate parent you would do path.at(-1)
state: State
— an object of the same type as the second argument to walk
. Visitors can pass new state objects to their children with next(childState)
or visit(node, childState)
next(state: State): void
— a function that allows you to control when child nodes are visited, and which state they are visited with. It can be called zero or one times — if zero, it will be called automatically once the current node has been visitedskip(): void
— prevents children from being visitedstop(): void
— prevents any subsequent traversalvisit(node: Node, state?: State): Node
— returns the result of visiting node
with the current set of visitors. If no state
is provided, children will inherit the current state
Immutability
ASTs are regarded as immutable. If you return a transformed node from a visitor, then all parents of the node will be replaced with clones, but unchanged subtrees will reuse the existing nodes.
For example in this case, no transformation takes place, meaning that the returned value is identical to the original AST:
const transformed = walk(original, state, {
Literal(node) {
console.log(node.value);
}
});
transformed === original;
In this case, however, we replace one of the nodes:
const original = {
type: 'BinaryExpression',
operator: '+',
left: {
type: 'Identifier',
name: 'foo'
},
left: {
type: 'Identifier',
name: 'bar'
}
};
const transformed = walk(original, state, {
Identifier(node) {
if (node.name === 'bar') {
return { ...node, name: 'baz' };
}
}
});
transformed === original;
transformed.left === original.left;
This makes it very easy to transform parts of your AST without incurring the performance and memory overhead of cloning the entire thing, and without the footgun of mutating it in place.
License
MIT