mirror of https://github.com/webpack/webpack.git
308 lines
8.6 KiB
JavaScript
308 lines
8.6 KiB
JavaScript
/*
|
|
MIT License http://www.opensource.org/licenses/mit-license.php
|
|
Author Tobias Koppers @sokra
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
const path = require("path");
|
|
const {
|
|
evaluateToIdentifier,
|
|
evaluateToString,
|
|
expressionIsUnsupported,
|
|
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");
|
|
|
|
/** @typedef {import("webpack-sources").ReplaceSource} ReplaceSource */
|
|
/** @typedef {import("./Compiler")} Compiler */
|
|
/** @typedef {import("./Dependency")} Dependency */
|
|
/** @typedef {import("./DependencyTemplates")} DependencyTemplates */
|
|
/** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
|
|
|
|
class NodeStuffPlugin {
|
|
constructor(options) {
|
|
this.options = options;
|
|
}
|
|
|
|
/**
|
|
* Apply the plugin
|
|
* @param {Compiler} compiler the compiler instance
|
|
* @returns {void}
|
|
*/
|
|
apply(compiler) {
|
|
const options = this.options;
|
|
compiler.hooks.compilation.tap(
|
|
"NodeStuffPlugin",
|
|
(compilation, { normalModuleFactory }) => {
|
|
compilation.dependencyFactories.set(
|
|
ModuleDecoratorDependency,
|
|
normalModuleFactory
|
|
);
|
|
compilation.dependencyTemplates.set(
|
|
ModuleDecoratorDependency,
|
|
new ModuleDecoratorDependency.Template()
|
|
);
|
|
|
|
compilation.hooks.runtimeRequirementInModule
|
|
.for(RuntimeGlobals.harmonyModuleDecorator)
|
|
.tap("NodeStuffPlugin", (module, set) => {
|
|
set.add(RuntimeGlobals.module);
|
|
set.add(RuntimeGlobals.require);
|
|
});
|
|
|
|
compilation.hooks.runtimeRequirementInModule
|
|
.for(RuntimeGlobals.nodeModuleDecorator)
|
|
.tap("NodeStuffPlugin", (module, set) => {
|
|
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()
|
|
);
|
|
});
|
|
|
|
const handler = (parser, parserOptions) => {
|
|
if (parserOptions.node === false) return;
|
|
|
|
let localOptions = options;
|
|
if (parserOptions.node) {
|
|
localOptions = Object.assign({}, localOptions, parserOptions.node);
|
|
}
|
|
|
|
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
|
|
);
|
|
dep.loc = expr.loc;
|
|
parser.state.current.addDependency(dep);
|
|
return true;
|
|
});
|
|
};
|
|
|
|
const setConstant = (expressionName, value) =>
|
|
setModuleConstant(expressionName, () => value);
|
|
|
|
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("?");
|
|
return evaluateToString(i < 0 ? resource : resource.substr(0, i))(
|
|
expr
|
|
);
|
|
});
|
|
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;
|
|
return evaluateToString(parser.state.module.context)(expr);
|
|
});
|
|
parser.hooks.expression
|
|
.for("require.main")
|
|
.tap(
|
|
"NodeStuffPlugin",
|
|
toConstantDependency(
|
|
parser,
|
|
`${RuntimeGlobals.moduleCache}[${
|
|
RuntimeGlobals.entryModuleId
|
|
}]`,
|
|
[RuntimeGlobals.moduleCache, RuntimeGlobals.entryModuleId]
|
|
)
|
|
);
|
|
parser.hooks.expression
|
|
.for("require.extensions")
|
|
.tap(
|
|
"NodeStuffPlugin",
|
|
expressionIsUnsupported(
|
|
parser,
|
|
"require.extensions is not supported by webpack. Use a loader instead."
|
|
)
|
|
);
|
|
parser.hooks.expression
|
|
.for("require.main.require")
|
|
.tap(
|
|
"NodeStuffPlugin",
|
|
expressionIsUnsupported(
|
|
parser,
|
|
"require.main.require is not supported by webpack."
|
|
)
|
|
);
|
|
parser.hooks.expression
|
|
.for("module.parent.require")
|
|
.tap(
|
|
"NodeStuffPlugin",
|
|
expressionIsUnsupported(
|
|
parser,
|
|
"module.parent.require is not supported by webpack."
|
|
)
|
|
);
|
|
parser.hooks.expression
|
|
.for("module.loaded")
|
|
.tap("NodeStuffPlugin", expr => {
|
|
parser.state.module.buildMeta.moduleConcatenationBailout =
|
|
"module.loaded";
|
|
return toConstantDependency(parser, "module.l", [
|
|
RuntimeGlobals.module
|
|
])(expr);
|
|
});
|
|
parser.hooks.expression
|
|
.for("module.id")
|
|
.tap("NodeStuffPlugin", expr => {
|
|
parser.state.module.buildMeta.moduleConcatenationBailout =
|
|
"module.id";
|
|
return toConstantDependency(parser, "module.i", [
|
|
RuntimeGlobals.module
|
|
])(expr);
|
|
});
|
|
parser.hooks.expression
|
|
.for("module.exports")
|
|
.tap("NodeStuffPlugin", expr => {
|
|
const module = parser.state.module;
|
|
const isHarmony =
|
|
module.buildMeta && module.buildMeta.exportsType;
|
|
if (!isHarmony) {
|
|
return toConstantDependency(parser, "module.exports", [
|
|
RuntimeGlobals.module
|
|
])(expr);
|
|
}
|
|
});
|
|
parser.hooks.evaluateIdentifier
|
|
.for("module.hot")
|
|
.tap("NodeStuffPlugin", evaluateToIdentifier("module.hot", false));
|
|
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
|
|
);
|
|
dep.loc = expr.loc;
|
|
parser.state.module.addDependency(dep);
|
|
return true;
|
|
});
|
|
};
|
|
|
|
normalModuleFactory.hooks.parser
|
|
.for("javascript/auto")
|
|
.tap("NodeStuffPlugin", handler);
|
|
normalModuleFactory.hooks.parser
|
|
.for("javascript/dynamic")
|
|
.tap("NodeStuffPlugin", handler);
|
|
}
|
|
);
|
|
}
|
|
}
|
|
|
|
class HarmonyModuleDecoratorRuntimeModule extends RuntimeModule {
|
|
constructor() {
|
|
super("harmony module decorator");
|
|
}
|
|
|
|
/**
|
|
* @returns {string} runtime code
|
|
*/
|
|
generate() {
|
|
return Template.asString([
|
|
`${RuntimeGlobals.harmonyModuleDecorator} = function(module) {`,
|
|
Template.indent([
|
|
"module = Object.create(module);",
|
|
"if (!module.children) module.children = [];",
|
|
"Object.defineProperty(module, 'loaded', {",
|
|
Template.indent([
|
|
"enumerable: true,",
|
|
"get: function () { return module.l; }"
|
|
]),
|
|
"});",
|
|
"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([
|
|
"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([
|
|
"enumerable: true,",
|
|
"get: function() { return module.i; }"
|
|
]),
|
|
"});",
|
|
"return module;"
|
|
]),
|
|
"};"
|
|
]);
|
|
}
|
|
}
|
|
|
|
module.exports = NodeStuffPlugin;
|