【发布时间】:2019-10-14 14:45:43
【问题描述】:
我正在尝试编写自定义 webpack 加载器。
Webpack 文档提到不推荐访问加载程序上下文中的 _compiler、_compilation 和 _module 属性 (https://webpack.js.org/api/loaders/#this_compilation),应该避免。
是否有更新的 API 可以让我访问它们?
【问题讨论】:
我正在尝试编写自定义 webpack 加载器。
Webpack 文档提到不推荐访问加载程序上下文中的 _compiler、_compilation 和 _module 属性 (https://webpack.js.org/api/loaders/#this_compilation),应该避免。
是否有更新的 API 可以让我访问它们?
【问题讨论】:
对于任何试图弄清楚同样事情的人。我认为没有任何替代 API。这个想法可能是加载器不应该真的需要访问这些对象,如果你确实需要它们,你应该使用插件来代替(或另外)。
如果这些属性将来会被删除,您可以使用插件“将它们重新添加”:
// custom-plugin.js
const NormalModule = require('webpack/lib/NormalModule')
class CustomPlugin {
apply (compiler) {
compiler.hooks.compilation.tap('CustomPlugin', (compilation) => {
NormalModule.getCompilationHooks(compilation).loader.tap('CustomPlugin', (loader, module) => {
loader._customPlugin = {
module: module,
compiler: compiler,
compilation: compilation,
}
})
});
}
}
module.exports = CustomPlugin
// custom-loader.js
module.exports = function (source) {
console.log({
module: this._customPlugin.module === this._module, // true
compiler: this._customPlugin.compiler === this._compiler, // true
compilation: this._customPlugin.compilation === this._compilation, // true
});
return source
}
【讨论】:
加载完依赖后,就可以访问生成模块的源码了:
// Index the modules generated in the child compiler by raw request.
let byRawRequest = new Map
for (let asset of compilation.modules)
byRawRequest.set(asset.rawRequest, asset)
// Replace the template requests with the result from modules generated in
// the child compiler.
for (let {node, request} of this._getAssetRequests()) {
if (!byRawRequest.has(request))
continue
const ASSET = byRawRequest.get(request)
const SOURCE = ASSET.originalSource().source()
const NEW_REQUEST = execAssetModule(SOURCE)
setResourceRequest(node, NEW_REQUEST)
log(`Changed: ${prettyFormat({from: request, to: NEW_REQUEST})}`)
}
And execute the module's source with a VM:
function execAssetModule(code, path) {
let script = new Script(code)
let exports = {}
let sandbox = {
__webpack_public_path__: '',
module: {exports},
exports,
}
script.runInNewContext(sandbox)
return sandbox.module.exports
}
【讨论】: