webpack/lib/NodeStuffPlugin.js

308 lines
8.6 KiB
JavaScript
Raw Normal View History

/*
MIT License http://www.opensource.org/licenses/mit-license.php
Author Tobias Koppers @sokra
*/
2018-07-30 23:08:51 +08:00
2017-02-22 23:34:10 +08:00
"use strict";
2017-02-22 23:34:10 +08:00
const path = require("path");
2018-07-03 16:24:29 +08:00
const {
evaluateToIdentifier,
evaluateToString,
expressionIsUnsupported,
2018-11-06 02:03:12 +08:00
toConstantDependency
} = require("./JavascriptParserHelpers");
const RuntimeGlobals = require("./RuntimeGlobals");
const RuntimeModule = require("./RuntimeModule");
const Template = require("./Template");
const CachedConstDependency = require("./dependencies/CachedConstDependency");
const ModuleDecoratorDependency = require("./dependencies/ModuleDecoratorDependency");
2018-07-27 21:57:03 +08:00
/** @typedef {import("webpack-sources").ReplaceSource} ReplaceSource */
2018-11-03 04:05:46 +08:00
/** @typedef {import("./Compiler")} Compiler */
2018-07-27 21:57:03 +08:00
/** @typedef {import("./Dependency")} Dependency */
/** @typedef {import("./DependencyTemplates")} DependencyTemplates */
/** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
2015-07-13 06:20:09 +08:00
2017-02-22 23:34:10 +08:00
class NodeStuffPlugin {
constructor(options) {
this.options = options;
}
2015-07-13 06:20:09 +08:00
2018-11-03 04:05:46 +08:00
/**
* Apply the plugin
* @param {Compiler} compiler the compiler instance
* @returns {void}
*/
2017-02-22 23:34:10 +08:00
apply(compiler) {
const options = this.options;
2018-02-25 09:00:20 +08:00
compiler.hooks.compilation.tap(
"NodeStuffPlugin",
(compilation, { normalModuleFactory }) => {
2018-07-27 21:57:03 +08:00
compilation.dependencyFactories.set(
ModuleDecoratorDependency,
normalModuleFactory
);
compilation.dependencyTemplates.set(
ModuleDecoratorDependency,
new ModuleDecoratorDependency.Template()
2018-07-27 21:57:03 +08:00
);
compilation.hooks.runtimeRequirementInModule
.for(RuntimeGlobals.harmonyModuleDecorator)
.tap("NodeStuffPlugin", (module, set) => {
2018-12-07 05:05:06 +08:00
set.add(RuntimeGlobals.module);
set.add(RuntimeGlobals.require);
});
compilation.hooks.runtimeRequirementInModule
.for(RuntimeGlobals.nodeModuleDecorator)
.tap("NodeStuffPlugin", (module, set) => {
2018-12-07 05:05:06 +08:00
set.add(RuntimeGlobals.module);
set.add(RuntimeGlobals.require);
});
compilation.hooks.runtimeRequirementInTree
.for(RuntimeGlobals.harmonyModuleDecorator)
.tap("NodeStuffPlugin", (chunk, set) => {
compilation.addRuntimeModule(
chunk,
new HarmonyModuleDecoratorRuntimeModule()
);
});
compilation.hooks.runtimeRequirementInTree
.for(RuntimeGlobals.nodeModuleDecorator)
.tap("NodeStuffPlugin", (chunk, set) => {
compilation.addRuntimeModule(
chunk,
new NodeModuleDecoratorRuntimeModule()
);
});
2018-02-25 09:00:20 +08:00
const handler = (parser, parserOptions) => {
if (parserOptions.node === false) return;
2017-02-22 23:34:10 +08:00
2018-02-25 09:00:20 +08:00
let localOptions = options;
if (parserOptions.node) {
2018-02-25 09:00:20 +08:00
localOptions = Object.assign({}, localOptions, parserOptions.node);
}
2018-02-25 09:00:20 +08:00
const setModuleConstant = (expressionName, fn) => {
parser.hooks.expression
.for(expressionName)
.tap("NodeStuffPlugin", expr => {
const dep = new CachedConstDependency(
JSON.stringify(fn(parser.state.module)),
expr.range,
expressionName
2018-02-25 09:00:20 +08:00
);
dep.loc = expr.loc;
parser.state.current.addDependency(dep);
2018-02-25 09:00:20 +08:00
return true;
});
};
const setConstant = (expressionName, value) =>
setModuleConstant(expressionName, () => value);
2018-02-25 09:00:20 +08:00
const context = compiler.context;
if (localOptions.__filename === "mock") {
setConstant("__filename", "/index.js");
} else if (localOptions.__filename) {
setModuleConstant("__filename", module =>
path.relative(context, module.resource)
);
}
parser.hooks.evaluateIdentifier
.for("__filename")
.tap("NodeStuffPlugin", expr => {
if (!parser.state.module) return;
const resource = parser.state.module.resource;
const i = resource.indexOf("?");
2018-07-03 16:24:29 +08:00
return evaluateToString(i < 0 ? resource : resource.substr(0, i))(
expr
);
2018-02-25 09:00:20 +08:00
});
if (localOptions.__dirname === "mock") {
setConstant("__dirname", "/");
} else if (localOptions.__dirname) {
setModuleConstant("__dirname", module =>
path.relative(context, module.context)
);
}
parser.hooks.evaluateIdentifier
.for("__dirname")
.tap("NodeStuffPlugin", expr => {
if (!parser.state.module) return;
2018-07-03 16:24:29 +08:00
return evaluateToString(parser.state.module.context)(expr);
2018-02-25 09:00:20 +08:00
});
parser.hooks.expression
.for("require.main")
.tap(
"NodeStuffPlugin",
2018-11-06 02:03:12 +08:00
toConstantDependency(
2018-02-25 09:00:20 +08:00
parser,
2018-11-06 02:03:12 +08:00
`${RuntimeGlobals.moduleCache}[${
RuntimeGlobals.entryModuleId
}]`,
[RuntimeGlobals.moduleCache, RuntimeGlobals.entryModuleId]
2018-02-25 09:00:20 +08:00
)
);
parser.hooks.expression
.for("require.extensions")
.tap(
"NodeStuffPlugin",
2018-07-03 16:24:29 +08:00
expressionIsUnsupported(
2018-02-25 09:00:20 +08:00
parser,
"require.extensions is not supported by webpack. Use a loader instead."
)
);
parser.hooks.expression
.for("require.main.require")
.tap(
"NodeStuffPlugin",
2018-07-23 18:19:16 +08:00
expressionIsUnsupported(
parser,
"require.main.require is not supported by webpack."
)
);
parser.hooks.expression
.for("module.parent.require")
.tap(
"NodeStuffPlugin",
2018-07-23 18:19:16 +08:00
expressionIsUnsupported(
parser,
"module.parent.require is not supported by webpack."
)
);
2018-02-25 09:00:20 +08:00
parser.hooks.expression
.for("module.loaded")
.tap("NodeStuffPlugin", expr => {
parser.state.module.buildMeta.moduleConcatenationBailout =
"module.loaded";
2018-11-17 01:18:44 +08:00
return toConstantDependency(parser, "module.l", [
RuntimeGlobals.module
])(expr);
2018-02-25 09:00:20 +08:00
});
parser.hooks.expression
.for("module.id")
.tap("NodeStuffPlugin", expr => {
parser.state.module.buildMeta.moduleConcatenationBailout =
"module.id";
2018-11-17 01:18:44 +08:00
return toConstantDependency(parser, "module.i", [
RuntimeGlobals.module
])(expr);
2018-02-25 09:00:20 +08:00
});
parser.hooks.expression
.for("module.exports")
2018-11-17 01:18:44 +08:00
.tap("NodeStuffPlugin", expr => {
2018-02-25 09:00:20 +08:00
const module = parser.state.module;
const isHarmony =
module.buildMeta && module.buildMeta.exportsType;
2018-11-17 01:18:44 +08:00
if (!isHarmony) {
return toConstantDependency(parser, "module.exports", [
RuntimeGlobals.module
])(expr);
}
2018-02-25 09:00:20 +08:00
});
parser.hooks.evaluateIdentifier
.for("module.hot")
2018-07-03 16:24:29 +08:00
.tap("NodeStuffPlugin", evaluateToIdentifier("module.hot", false));
2018-07-27 21:57:03 +08:00
parser.hooks.expression.for("module").tap("NodeStuffPlugin", expr => {
const isHarmony =
parser.state.module.buildMeta &&
parser.state.module.buildMeta.exportsType;
const dep = new ModuleDecoratorDependency(
isHarmony
? RuntimeGlobals.harmonyModuleDecorator
: RuntimeGlobals.nodeModuleDecorator
2018-02-25 09:00:20 +08:00
);
2018-07-27 21:57:03 +08:00
dep.loc = expr.loc;
parser.state.module.addDependency(dep);
return true;
2017-02-22 23:34:10 +08:00
});
2017-11-08 18:32:05 +08:00
};
2018-02-25 09:00:20 +08:00
normalModuleFactory.hooks.parser
.for("javascript/auto")
.tap("NodeStuffPlugin", handler);
normalModuleFactory.hooks.parser
.for("javascript/dynamic")
.tap("NodeStuffPlugin", handler);
}
);
2017-02-22 23:34:10 +08:00
}
}
2018-07-27 21:57:03 +08:00
class HarmonyModuleDecoratorRuntimeModule extends RuntimeModule {
constructor() {
super("harmony module decorator");
}
/**
* @returns {string} runtime code
*/
generate() {
return Template.asString([
`${RuntimeGlobals.harmonyModuleDecorator} = function(module) {`,
Template.indent([
2018-12-07 05:05:06 +08:00
"module = Object.create(module);",
"if (!module.children) module.children = [];",
"Object.defineProperty(module, 'loaded', {",
Template.indent([
2018-12-07 05:05:06 +08:00
"enumerable: true,",
"get: function () { return module.l; }"
]),
2018-12-07 05:05:06 +08:00
"});",
"Object.defineProperty(module, 'id', {",
Template.indent([
"enumerable: true,",
"get: function () { return module.i; }"
]),
"});",
"Object.defineProperty(module, 'exports', {",
Template.indent("enumerable: true"),
"});",
"return module;"
]),
"};"
]);
}
}
class NodeModuleDecoratorRuntimeModule extends RuntimeModule {
constructor() {
super("node module decorator");
}
/**
* @returns {string} runtime code
*/
generate() {
return Template.asString([
`${RuntimeGlobals.nodeModuleDecorator} = function(module) {`,
Template.indent([
2018-12-07 05:05:06 +08:00
"module.paths = [];",
"if (!module.children) module.children = [];",
"Object.defineProperty(module, 'loaded', {",
Template.indent([
"enumerable: true,",
"get: function() { return module.l; }"
]),
"});",
"Object.defineProperty(module, 'id', {",
Template.indent([
2018-12-07 05:05:06 +08:00
"enumerable: true,",
"get: function() { return module.i; }"
]),
2018-12-07 05:05:06 +08:00
"});",
"return module;"
]),
"};"
]);
}
}
2017-02-22 23:34:10 +08:00
module.exports = NodeStuffPlugin;