Skip to content

Instantly share code, notes, and snippets.

@matthewma7
Created January 13, 2021 16:28
Show Gist options
  • Star 9 You must be signed in to star a gist
  • Fork 4 You must be signed in to fork a gist
  • Save matthewma7/915b28846ad99d582e6b4ddea6dfc309 to your computer and use it in GitHub Desktop.
Save matthewma7/915b28846ad99d582e6b4ddea6dfc309 to your computer and use it in GitHub Desktop.
/*!
* ExternalTemplateRemotesPlugin
* License: MIT (https://mit-license.org/)
*/
const extractUrlAndGlobal = require('webpack/lib/util/extractUrlAndGlobal');
const { RawSource } = require('webpack-sources');
const PLUGIN_NAME = 'ExternalTemplateRemotesPlugin';
class ExternalTemplateRemotesPlugin {
apply(compiler) {
compiler.hooks.make.tap(PLUGIN_NAME, compilation => {
const scriptExternalModules = [];
compilation.hooks.buildModule.tap(PLUGIN_NAME, module => {
if (module.constructor.name === 'ExternalModule' && module.externalType === 'script') {
scriptExternalModules.push(module);
}
});
compilation.hooks.afterCodeGeneration.tap(PLUGIN_NAME, function() {
scriptExternalModules.map(module => {
const urlTemplate = extractUrlAndGlobal(module.request)[0];
const urlExpression = toExpression(urlTemplate);
const sourceMap = compilation.codeGenerationResults.get(module).sources;
const rawSource = sourceMap.get('javascript');
sourceMap.set(
'javascript',
new RawSource(rawSource.source().replace(`"${urlTemplate}"`, urlExpression))
);
});
});
});
}
}
function toExpression(templateUrl) {
const result = [];
const current = [];
let isExpression = false;
let invalid = false;
for (const c of templateUrl) {
if (c === '[') {
if (isExpression) {
invalid = true;
break;
}
isExpression = true;
if (current.length) {
result.push(`"${current.join('')}"`);
current.length = 0;
}
} else if (c === ']') {
if (!isExpression) {
invalid = true;
break;
}
isExpression = false;
if (current.length) {
result.push(`${current.join('')}`);
current.length = 0;
}
current.length = 0;
} else {
current.push(c);
}
}
if (isExpression || invalid) {
throw new Error(`Invalid template URL "${templateUrl}"`);
}
if (current.length) {
result.push(`"${current.join('')}"`);
}
return result.join(' + ');
}
module.exports = ExternalTemplateRemotesPlugin;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment