The parser instance, found in the compiler, is used to parse each module being processed by webpack. The parser is yet another webpack class that extends tapable and provides a variety of tapable hooks that can be used by plugin authors to customize the parsing process.
The parser is found within NormalModuleFactory and therefore takes little more work to access:
compiler.hooks.normalModuleFactory.tap('MyPlugin', (factory) => {
factory.hooks.parser
.for('javascript/auto')
.tap('MyPlugin', (parser, options) => {
parser.hooks.someHook.tap(/* ... */);
});
}); As with the compiler, tapAsync and tapPromise may also be available depending on the type of hook.
The following lifecycle hooks are exposed by the parser and can be accessed as such:
SyncBailHook
Triggered when evaluating an expression consisting in a typeof of a free variable
identifier
expression
parser.hooks.evaluateTypeof
.for('myIdentifier')
.tap('MyPlugin', (expression) => {
/* ... */
return expressionResult;
}); This will trigger the evaluateTypeof hook:
const a = typeof myIdentifier;
This won't trigger:
const myIdentifier = 0; const b = typeof myIdentifier;
SyncBailHook
Called when evaluating an expression.
expressionType
expression
For example:
index.js
const a = new String();
MyPlugin.js
parser.hooks.evaluate.for('NewExpression').tap('MyPlugin', (expression) => {
/* ... */
return expressionResult;
}); Where the expressions types are:
'ArrowFunctionExpression''AssignmentExpression''AwaitExpression''BinaryExpression''CallExpression''ClassExpression''ConditionalExpression''FunctionExpression''Identifier''LogicalExpression''MemberExpression''NewExpression''ObjectExpression''SequenceExpression''SpreadElement''TaggedTemplateExpression''TemplateLiteral''ThisExpression''UnaryExpression''UpdateExpression'SyncBailHook
Called when evaluating an identifier that is a free variable.
identifier
expression
SyncBailHook
Called when evaluating an identifier that is a defined variable.
identifier
expression
SyncBailHook
Called when evaluating a call to a member function of a successfully evaluated expression.
identifier
expression param
This expression will trigger the hook:
index.js
const a = expression.myFunc();
MyPlugin.js
parser.hooks.evaluateCallExpressionMember
.for('myFunc')
.tap('MyPlugin', (expression, param) => {
/* ... */
return expressionResult;
}); SyncBailHook
General purpose hook that is called for every parsed statement in a code fragment.
statement
parser.hooks.statement.tap('MyPlugin', (statement) => {
/* ... */
}); Where the statement.type could be:
'BlockStatement''VariableDeclaration''FunctionDeclaration''ReturnStatement''ClassDeclaration''ExpressionStatement''ImportDeclaration''ExportAllDeclaration''ExportDefaultDeclaration''ExportNamedDeclaration''IfStatement''SwitchStatement''ForInStatement''ForOfStatement''ForStatement''WhileStatement''DoWhileStatement''ThrowStatement''TryStatement''LabeledStatement''WithStatement'SyncBailHook
Called when parsing an if statement. Same as the statement hook, but triggered only when statement.type == 'IfStatement'.
statement
SyncBailHook
Called when parsing statements with a label. Those statements have statement.type === 'LabeledStatement'.
labelName
statement
SyncBailHook
Called for every import statement in a code fragment. The source parameter contains the name of the imported file.
statement source
The following import statement will trigger the hook once:
index.js
import _ from 'lodash';
MyPlugin.js
parser.hooks.import.tap('MyPlugin', (statement, source) => {
// source == 'lodash'
}); SyncBailHook
Called for every specifier of every import statement.
statement source exportName identifierName
The following import statement will trigger the hook twice:
index.js
import _, { has } from 'lodash'; MyPlugin.js
parser.hooks.importSpecifier.tap(
'MyPlugin',
(statement, source, exportName, identifierName) => {
/* First call
source == 'lodash'
exportName == 'default'
identifierName == '_'
*/
/* Second call
source == 'lodash'
exportName == 'has'
identifierName == 'has'
*/
}
); SyncBailHook
Called for every export statement in a code fragment.
statement
SyncBailHook
Called for every export-import statement eg: export * from 'otherModule';.
statement source
SyncBailHook
Called for every export statement exporting a declaration.
statement declaration
Those exports will trigger this hook:
export const myVar = 'hello'; // also var, let
export function FunctionName() {}
export class ClassName {} SyncBailHook
Called for every export statement exporting an expression e.g.export default expression;.
statement declaration
SyncBailHook
Called for every specifier of every export statement.
statement identifierName exportName index
SyncBailHook
Called for every specifier of every export-import statement.
statement source identifierName exportName index
SyncBailHook
Called when parsing a variable declaration.
declaration
SyncBailHook
Called when parsing a variable declaration defined using let
declaration
SyncBailHook
Called when parsing a variable declaration defined using const
declaration
SyncBailHook
Called when parsing a variable declaration defined using var
declaration
SyncBailHook
Triggered before renaming an identifier to determine if the renaming is allowed. This is usually used together with the rename hook.
identifier
expression
var a = b;
parser.hooks.canRename.for('b').tap('MyPlugin', (expression) => {
// returning true allows renaming
return true;
}); SyncBailHook
Triggered when renaming to get the new identifier. This hook will be called only if canRename returns true.
identifier
expression
var a = b;
parser.hooks.rename.for('b').tap('MyPlugin', (expression) => {}); SyncBailHook
Called when parsing an AssignmentExpression before parsing the assigned expression.
identifier
expression
a += b;
parser.hooks.assigned.for('a').tap('MyPlugin', (expression) => {
// this is called before parsing b
}); SyncBailHook
Called when parsing an AssignmentExpression before parsing the assign expression.
identifier
expression
a += b;
parser.hooks.assigned.for('a').tap('MyPlugin', (expression) => {
// this is called before parsing a
}); SyncBailHook
Triggered when parsing the typeof of an identifier
identifier
expression
SyncBailHook
Called when parsing a function call.
identifier
expression
eval(/* something */);
parser.hooks.call.for('eval').tap('MyPlugin', (expression) => {}); SyncBailHook
Triggered when parsing a call to a member function of an object.
objectIdentifier
expression, properties
myObj.anyFunc();
parser.hooks.callMemberChain
.for('myObj')
.tap('MyPlugin', (expression, properties) => {}); SyncBailHook
Invoked when parsing a new expression.
identifier
expression
new MyClass();
parser.hooks.new.for('MyClass').tap('MyPlugin', (expression) => {}); SyncBailHook
Called when parsing an expression.
identifier
expression
const a = this;
parser.hooks.expression.for('this').tap('MyPlugin', (expression) => {}); SyncBailHook
Called when parsing a ConditionalExpression e.g. condition ? a : b
expression
SyncBailHook
Get access to the abstract syntax tree (AST) of a code fragment
ast comments
© JS Foundation and other contributors
Licensed under the Creative Commons Attribution License 4.0.
https://webpack.js.org/api/parser