Skip to content

Commit 8845e26

Browse files
author
jeromeh
committed
feat: support of injectHot and injectClient on specific chunks
1 parent b38c257 commit 8845e26

File tree

5 files changed

+176
-37
lines changed

5 files changed

+176
-37
lines changed

lib/options.json

+16-2
Original file line numberDiff line numberDiff line change
@@ -228,6 +228,13 @@
228228
{
229229
"type": "boolean"
230230
},
231+
{
232+
"type": "array",
233+
"items": {
234+
"type": "string"
235+
},
236+
"minItems": 1
237+
},
231238
{
232239
"instanceof": "Function"
233240
}
@@ -238,6 +245,13 @@
238245
{
239246
"type": "boolean"
240247
},
248+
{
249+
"type": "array",
250+
"items": {
251+
"type": "string"
252+
},
253+
"minItems": 1
254+
},
241255
{
242256
"instanceof": "Function"
243257
}
@@ -395,8 +409,8 @@
395409
"hot": "should be {Boolean|String} (https://webpack.js.org/configuration/dev-server/#devserverhot)",
396410
"http2": "should be {Boolean} (https://webpack.js.org/configuration/dev-server/#devserverhttp2)",
397411
"https": "should be {Object|Boolean} (https://webpack.js.org/configuration/dev-server/#devserverhttps)",
398-
"injectClient": "should be {Boolean|Function} (https://webpack.js.org/configuration/dev-server/#devserverinjectclient)",
399-
"injectHot": "should be {Boolean|Function} (https://webpack.js.org/configuration/dev-server/#devserverinjecthot)",
412+
"injectClient": "should be {Boolean|String[]|Function} (https://webpack.js.org/configuration/dev-server/#devserverinjectclient)",
413+
"injectHot": "should be {Boolean|String[]|Function} (https://webpack.js.org/configuration/dev-server/#devserverinjecthot)",
400414
"liveReload": "should be {Boolean} (https://webpack.js.org/configuration/dev-server/#devserverlivereload)",
401415
"onAfterSetupMiddleware": "should be {Function} (https://webpack.js.org/configuration/dev-server/#devserverafter)",
402416
"onBeforeSetupMiddleware": "should be {Function} (https://webpack.js.org/configuration/dev-server/#devserverbefore)",

lib/utils/DevServerPlugin.js

+92-28
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,13 @@ class DevServerPlugin {
2020
* @typedef {(string[] | string | Object<string | string[],string>)} Entry
2121
*/
2222

23+
/**
24+
* Additional entry to add to specific chunk
25+
* @typedef {Object} AdditionalChunkEntry
26+
* @property {Entry} entry
27+
* @property {string[]} [chunks]
28+
*/
29+
2330
/**
2431
* Apply the plugin
2532
* @param {Object} compiler the compiler instance
@@ -66,7 +73,7 @@ class DevServerPlugin {
6673
/**
6774
* prependEntry Method for webpack 4
6875
* @param {Entry} originalEntry
69-
* @param {Entry} additionalEntries
76+
* @param {AdditionalChunkEntry[]} additionalEntries
7077
* @returns {Entry}
7178
*/
7279
const prependEntry = (originalEntry, additionalEntries) => {
@@ -83,8 +90,13 @@ class DevServerPlugin {
8390

8491
Object.keys(originalEntry).forEach((key) => {
8592
// entry[key] should be a string here
93+
const chunkAdditionalEntries = additionalEntries.filter(
94+
(additionalEntry) =>
95+
!additionalEntry.chunks || additionalEntry.chunks.includes(key)
96+
);
97+
8698
const entryDescription = originalEntry[key];
87-
clone[key] = prependEntry(entryDescription, additionalEntries);
99+
clone[key] = prependEntry(entryDescription, chunkAdditionalEntries);
88100
});
89101

90102
return clone;
@@ -93,13 +105,15 @@ class DevServerPlugin {
93105
// in this case, entry is a string or an array.
94106
// make sure that we do not add duplicates.
95107
/** @type {Entry} */
96-
const entriesClone = additionalEntries.slice(0);
108+
const newEntries = additionalEntries.map(
109+
(additionalEntry) => additionalEntry.entry
110+
);
97111
[].concat(originalEntry).forEach((newEntry) => {
98-
if (!entriesClone.includes(newEntry)) {
99-
entriesClone.push(newEntry);
112+
if (!newEntries.includes(newEntry)) {
113+
newEntries.push(newEntry);
100114
}
101115
});
102-
return entriesClone;
116+
return newEntries;
103117
};
104118

105119
/**
@@ -112,14 +126,15 @@ class DevServerPlugin {
112126

113127
/**
114128
*
115-
* @param {Boolean | checkInjectOptionsParam} option - inject(Hot|Client) it is Boolean | fn => Boolean
129+
* @param {Boolean | string[] | checkInjectOptionsParam} option - inject(Hot|Client) it is Boolean | fn => Boolean
116130
* @param {Object} _config
117131
* @param {Boolean} defaultValue
118-
* @return {Boolean}
132+
* @return {Boolean | string[]}
119133
*/
120134
// eslint-disable-next-line no-shadow
121135
const checkInject = (option, _config, defaultValue) => {
122136
if (typeof option === 'boolean') return option;
137+
if (Array.isArray(option)) return option;
123138
if (typeof option === 'function') return option(_config);
124139
return defaultValue;
125140
};
@@ -138,37 +153,86 @@ class DevServerPlugin {
138153
undefined,
139154
null,
140155
].includes(compilerOptions.target);
141-
/** @type {Entry} */
142-
const additionalEntries = checkInject(
156+
/** @type {AdditionalChunkEntry[]} */
157+
const additionalEntries = [];
158+
159+
const checkInjectClientResult = checkInject(
143160
options.injectClient,
144161
compilerOptions,
145162
isWebTarget
146-
)
147-
? [clientEntry]
148-
: [];
163+
);
164+
if (checkInjectClientResult) {
165+
additionalEntries.push({
166+
entry: clientEntry,
167+
chunks: Array.isArray(checkInjectClientResult)
168+
? checkInjectClientResult
169+
: null,
170+
});
171+
}
149172

150-
if (hotEntry && checkInject(options.injectHot, compilerOptions, true)) {
151-
additionalEntries.push(hotEntry);
173+
if (hotEntry) {
174+
const checkInjectHotResult = checkInject(
175+
options.injectHot,
176+
compilerOptions,
177+
true
178+
);
179+
if (checkInjectHotResult) {
180+
additionalEntries.push({
181+
entry: hotEntry,
182+
chunks: Array.isArray(checkInjectHotResult)
183+
? checkInjectHotResult
184+
: null,
185+
});
186+
}
152187
}
153188

154189
// use a hook to add entries if available
155190
if (EntryPlugin) {
156191
compiler.hooks.make.tapPromise('DevServerPlugin', (compilation) =>
157192
Promise.all(
158-
additionalEntries.map(
159-
(entry) =>
160-
new Promise((resolve, reject) => {
161-
compilation.addEntry(
162-
compiler.context,
163-
EntryPlugin.createDependency(entry, {}),
164-
{}, // global entry
165-
(err) => {
166-
if (err) return reject(err);
167-
resolve();
168-
}
193+
additionalEntries.map((additionalChunkEntry) => {
194+
// add entry to existing chunks
195+
if (
196+
additionalChunkEntry.chunks &&
197+
Array.isArray(additionalChunkEntry.chunks)
198+
) {
199+
let promise = Promise.resolve();
200+
additionalChunkEntry.chunks.forEach((chunkName) => {
201+
promise = promise.then(
202+
() =>
203+
new Promise((resolve, reject) => {
204+
compilation.addEntry(
205+
compiler.context,
206+
EntryPlugin.createDependency(
207+
additionalChunkEntry.entry,
208+
{}
209+
),
210+
chunkName,
211+
(err) => {
212+
if (err) return reject(err);
213+
resolve();
214+
}
215+
);
216+
})
169217
);
170-
})
171-
)
218+
});
219+
220+
return promise;
221+
}
222+
223+
// add new entry
224+
return new Promise((resolve, reject) => {
225+
compilation.addEntry(
226+
compiler.context,
227+
EntryPlugin.createDependency(additionalChunkEntry.entry, {}),
228+
{}, // global entry
229+
(err) => {
230+
if (err) return reject(err);
231+
resolve();
232+
}
233+
);
234+
});
235+
})
172236
)
173237
);
174238
} else {

test/__snapshots__/Validation.test.js.snap

+3-1
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,11 @@ exports[`Validation validation should fail validation for invalid \`hot\` config
1212
exports[`Validation validation should fail validation for invalid \`injectHot\` configuration 1`] = `
1313
"Invalid configuration object. Object has been initialized using a configuration object that does not match the API schema.
1414
- configuration.injectHot should be one of these:
15-
boolean | function
15+
boolean | [string, ...] (should not have fewer than 1 item) | function
1616
Details:
1717
* configuration.injectHot should be a boolean.
18+
* configuration.injectHot should be an array:
19+
[string, ...] (should not have fewer than 1 item)
1820
* configuration.injectHot should be an instance of function."
1921
`;
2022

test/options.test.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -359,11 +359,11 @@ describe('options', () => {
359359
],
360360
},
361361
injectClient: {
362-
success: [true, () => {}],
362+
success: [true, ['a'], () => {}],
363363
failure: [''],
364364
},
365365
injectHot: {
366-
success: [true, () => {}],
366+
success: [true, ['a'], () => {}],
367367
failure: [''],
368368
},
369369
onListening: {

test/server/utils/DevServerPlugin.test.js

+63-4
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,22 @@ describe('DevServerPlugin util', () => {
1717
const entries = [];
1818

1919
const compilation = {
20-
addEntry(_context, dep, _options, cb) {
20+
addEntry(_context, dep, _optionsOrName, cb) {
2121
if (!dep.loc.name) {
22-
entries.push(dep.request);
22+
const name =
23+
typeof _optionsOrName === 'object'
24+
? _optionsOrName.name
25+
: _optionsOrName;
26+
27+
if (name && entryOption[name]) {
28+
const entry = entryOption[name];
29+
entries[name] = {
30+
...entry,
31+
import: [dep.request, ...entry.import],
32+
};
33+
} else {
34+
entries.push(dep.request);
35+
}
2336
}
2437
cb();
2538
},
@@ -39,8 +52,7 @@ describe('DevServerPlugin util', () => {
3952
entries.push(...entryOption.main.import);
4053
}
4154
// merge named exports into entries
42-
Object.assign(entries, entryOption);
43-
return entries;
55+
return Object.assign([], entryOption, entries);
4456
}
4557
return entryOption;
4658
}
@@ -621,6 +633,53 @@ describe('DevServerPlugin util', () => {
621633
);
622634
});
623635

636+
it('should allows selecting chunks to inline the client into', async () => {
637+
const webpackOptions = [
638+
Object.assign({}, config, {
639+
entry: {
640+
chunk1: ['./foo.js'],
641+
chunk2: './foo.js',
642+
chunk3: './foo.js',
643+
},
644+
}),
645+
];
646+
const compiler = webpack(webpackOptions);
647+
648+
const devServerOptions = {
649+
injectClient: ['chunk1', 'chunk3'],
650+
transportMode: {
651+
server: 'sockjs',
652+
client: 'sockjs',
653+
},
654+
};
655+
656+
await Promise.all(
657+
// eslint-disable-next-line no-shadow
658+
compiler.compilers.map((compiler) => {
659+
const plugin = new DevServerPlugin(devServerOptions);
660+
plugin.apply(compiler);
661+
return getEntries(compiler).then((entries) => {
662+
expect(Object.keys(entries).length).toEqual(3);
663+
expect(entries.chunk1.import.length).toEqual(2);
664+
expect(entries.chunk2.import.length).toEqual(1);
665+
expect(entries.chunk3.import.length).toEqual(2);
666+
667+
expect(
668+
normalize(entries.chunk1.import[0]).indexOf(
669+
'client/default/index.js?'
670+
) !== -1
671+
).toBeTruthy();
672+
expect(normalize(entries.chunk2.import[0])).toEqual('./foo.js');
673+
expect(
674+
normalize(entries.chunk3.import[0]).indexOf(
675+
'client/default/index.js?'
676+
) !== -1
677+
).toBeTruthy();
678+
});
679+
})
680+
);
681+
});
682+
624683
it('should prepends the hot runtime to all targets by default (when hot)', async () => {
625684
const webpackOptions = [
626685
Object.assign({ target: 'web' }, config),

0 commit comments

Comments
 (0)