-
Notifications
You must be signed in to change notification settings - Fork 618
/
Copy pathSetPublicPathCurrentScriptPlugin.ts
113 lines (96 loc) · 4.16 KB
/
SetPublicPathCurrentScriptPlugin.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license.
// See LICENSE in the project root for license information.
import type webpack from 'webpack';
import { SetPublicPathPluginBase } from './SetPublicPathPluginBase';
const PLUGIN_NAME: string = 'set-webpack-public-path-current-script-plugin';
const CURRENT_SCRIPT_VARIABLE_NAME: string = '__RUSHSTACK_CURRENT_SCRIPT__';
const PUBLIC_PATH_VARIABLE_NAME: string = '_publicPath';
type JavascriptModulesPluginHooks = ReturnType<
typeof webpack.javascript.JavascriptModulesPlugin.getCompilationHooks
>;
type CodeGenerationResults = Parameters<
Parameters<JavascriptModulesPluginHooks['render']['tap']>[1]
>[1]['codeGenerationResults'];
/**
* This simple plugin wraps the webpack bundle in an IIFE that sets a the `document.currentScript` value to a variable
* that is then used to populate the `__webpack_public_path__` variable.
*
* @public
*/
export class SetPublicPathCurrentScriptPlugin extends SetPublicPathPluginBase {
public constructor() {
super(PLUGIN_NAME);
}
protected _applyCompilation(thisWebpack: typeof webpack, compilation: webpack.Compilation): void {
const outputLibraryType: string | undefined = compilation.options.output.library?.type;
class SetPublicPathRuntimeModule extends thisWebpack.RuntimeModule {
public constructor() {
super('publicPath', thisWebpack.RuntimeModule.STAGE_BASIC);
}
public generate(): string {
return [
`var ${PUBLIC_PATH_VARIABLE_NAME} = ${CURRENT_SCRIPT_VARIABLE_NAME} ? ${CURRENT_SCRIPT_VARIABLE_NAME}.src : '';`,
`${thisWebpack.RuntimeGlobals.publicPath} = ${PUBLIC_PATH_VARIABLE_NAME}.slice(0, ${PUBLIC_PATH_VARIABLE_NAME}.lastIndexOf('/') + 1);`
].join('\n');
}
}
const runtimeModule: SetPublicPathRuntimeModule = new SetPublicPathRuntimeModule();
function appliesToChunk(chunk: webpack.Chunk, codeGenerationResults: CodeGenerationResults): boolean {
return chunk.hasRuntime() && codeGenerationResults.has(runtimeModule, chunk.runtime);
}
compilation.hooks.runtimeRequirementInTree
.for(thisWebpack.RuntimeGlobals.publicPath)
.tap(PLUGIN_NAME, (chunk: webpack.Chunk, set: Set<string>) => {
compilation.addRuntimeModule(chunk, runtimeModule);
});
const javascriptModulesPluginHooks: JavascriptModulesPluginHooks =
thisWebpack.javascript.JavascriptModulesPlugin.getCompilationHooks(compilation);
javascriptModulesPluginHooks.render.tap(
{ name: PLUGIN_NAME, stage: Number.MAX_SAFE_INTEGER },
(source, { codeGenerationResults, chunk }) => {
if (appliesToChunk(chunk, codeGenerationResults)) {
return new thisWebpack.sources.ConcatSource(
`(()=>{ var ${CURRENT_SCRIPT_VARIABLE_NAME} = document.currentScript; `,
source,
'})();'
);
} else {
return source;
}
}
);
javascriptModulesPluginHooks.chunkHash.tap(PLUGIN_NAME, (chunk, hash, { codeGenerationResults }) => {
hash.update(PLUGIN_NAME);
if (appliesToChunk(chunk, codeGenerationResults)) {
hash.update('set-public-path');
}
});
compilation.hooks.afterSeal.tap(PLUGIN_NAME, () => {
let hasProblematicLibraryType: boolean = false;
switch (outputLibraryType) {
case 'var':
case 'module':
hasProblematicLibraryType = true;
break;
}
if (hasProblematicLibraryType) {
const codeGenerationResults: CodeGenerationResults = compilation.codeGenerationResults;
let appliesToAnyChunk: boolean = false;
for (const chunk of compilation.chunks) {
if (appliesToChunk(chunk, codeGenerationResults)) {
appliesToAnyChunk = true;
break;
}
}
if (appliesToAnyChunk) {
compilation.errors.push(
new thisWebpack.WebpackError(
`The "${outputLibraryType}" output.library.type is not supported by the ${SetPublicPathCurrentScriptPlugin.name}` +
' plugin. Including this plugin with produce unexpected or invalid results.'
)
);
}
}
});
}
}