Skip to content

Commit 35dac70

Browse files
feat: added modifyResponseData option (#1529)
1 parent f48e442 commit 35dac70

10 files changed

+234
-77
lines changed

README.md

+43-22
Original file line numberDiff line numberDiff line change
@@ -60,18 +60,19 @@ See [below](#other-servers) for an example of use with fastify.
6060

6161
## Options
6262

63-
| Name | Type | Default | Description |
64-
| :-----------------------------------------: | :--------: | :-------------------------------------------: | :--------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------------------------------------------- |
65-
| **[`methods`](#methods)** | `Array` | `[ 'GET', 'HEAD' ]` | Allows to pass the list of HTTP request methods accepted by the middleware |
66-
| **[`headers`](#headers)** | `Array\ | Object\| Function` | `undefined` | Allows to pass custom HTTP headers on each request. |
67-
| **[`index`](#index)** | `Boolean\ | String` | `index.html` | If `false` (but not `undefined`), the server will not respond to requests to the root URL. |
68-
| **[`mimeTypes`](#mimetypes)** | `Object` | `undefined` | Allows to register custom mime types or extension mappings. |
69-
| **[`mimeTypeDefault`](#mimetypedefault)** | `String` | `undefined` | Allows to register a default mime type when we can't determine the content type. |
70-
| **[`publicPath`](#publicpath)** | `String` | `output.publicPath` (from a configuration) | The public path that the middleware is bound to. |
71-
| **[`stats`](#stats)** | `Boolean\ | String\| Object` | `stats` (from a configuration) | Stats options object or preset name. |
72-
| **[`serverSideRender`](#serversiderender)** | `Boolean` | `undefined` | Instructs the module to enable or disable the server-side rendering mode. |
73-
| **[`writeToDisk`](#writetodisk)** | `Boolean\ | Function` | `false` | Instructs the module to write files to the configured location on disk as specified in your `webpack` configuration. |
74-
| **[`outputFileSystem`](#outputfilesystem)** | `Object` | [`memfs`](https://github.com/streamich/memfs) | Set the default file system which will be used by webpack as primary destination of generated files. |
63+
| Name | Type | Default | Description |
64+
| :---------------------------------------------: | :-----------------------: | :-------------------------------------------: | :------------------------------------------------------------------------------------------------------------------- |
65+
| **[`methods`](#methods)** | `Array` | `[ 'GET', 'HEAD' ]` | Allows to pass the list of HTTP request methods accepted by the middleware |
66+
| **[`headers`](#headers)** | `Array\|Object\|Function` | `undefined` | Allows to pass custom HTTP headers on each request. |
67+
| **[`index`](#index)** | `Boolean\|String` | `index.html` | If `false` (but not `undefined`), the server will not respond to requests to the root URL. |
68+
| **[`mimeTypes`](#mimetypes)** | `Object` | `undefined` | Allows to register custom mime types or extension mappings. |
69+
| **[`mimeTypeDefault`](#mimetypedefault)** | `String` | `undefined` | Allows to register a default mime type when we can't determine the content type. |
70+
| **[`publicPath`](#publicpath)** | `String` | `output.publicPath` (from a configuration) | The public path that the middleware is bound to. |
71+
| **[`stats`](#stats)** | `Boolean\|String\|Object` | `stats` (from a configuration) | Stats options object or preset name. |
72+
| **[`serverSideRender`](#serversiderender)** | `Boolean` | `undefined` | Instructs the module to enable or disable the server-side rendering mode. |
73+
| **[`writeToDisk`](#writetodisk)** | `Boolean\|Function` | `false` | Instructs the module to write files to the configured location on disk as specified in your `webpack` configuration. |
74+
| **[`outputFileSystem`](#outputfilesystem)** | `Object` | [`memfs`](https://github.com/streamich/memfs) | Set the default file system which will be used by webpack as primary destination of generated files. |
75+
| **[`modifyResponseData`](#modifyresponsedata)** | `Function` | `undefined` | Allows to set up a callback to change the response data. |
7576

7677
The middleware accepts an `options` Object. The following is a property reference for the Object.
7778

@@ -119,14 +120,13 @@ webpackDevMiddleware(compiler, {
119120
headers: [
120121
{
121122
key: "X-custom-header",
122-
value: "foo"
123+
value: "foo",
123124
},
124125
{
125126
key: "Y-custom-header",
126-
value: "bar"
127-
}
128-
]
129-
},
127+
value: "bar",
128+
},
129+
],
130130
});
131131
```
132132

@@ -137,14 +137,13 @@ webpackDevMiddleware(compiler, {
137137
headers: () => [
138138
{
139139
key: "X-custom-header",
140-
value: "foo"
140+
value: "foo",
141141
},
142142
{
143143
key: "Y-custom-header",
144-
value: "bar"
145-
}
146-
]
147-
},
144+
value: "bar",
145+
},
146+
],
148147
});
149148
```
150149

@@ -250,6 +249,28 @@ const compiler = webpack({
250249
middleware(compiler, { outputFileSystem: myOutputFileSystem });
251250
```
252251

252+
### modifyResponseData
253+
254+
Allows to set up a callback to change the response data.
255+
256+
```js
257+
const webpack = require("webpack");
258+
const configuration = {
259+
/* Webpack configuration */
260+
};
261+
const compiler = webpack(configuration);
262+
263+
middleware(compiler, {
264+
// Note - if you send the `Range` header you will have `ReadStream`
265+
// Also `data` can be `string` or `Buffer`
266+
modifyResponseData: (req, res, data, byteLength) => {
267+
// Your logic
268+
// Don't use `res.end()` or `res.send()` here
269+
return { data, byteLength };
270+
},
271+
});
272+
```
273+
253274
## API
254275

255276
`webpack-dev-middleware` also provides convenience methods that can be use to

src/index.js

+19
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ const noop = () => {};
1717
/** @typedef {import("webpack").Configuration} Configuration */
1818
/** @typedef {import("webpack").Stats} Stats */
1919
/** @typedef {import("webpack").MultiStats} MultiStats */
20+
/** @typedef {import("fs").ReadStream} ReadStream */
2021

2122
/**
2223
* @typedef {Object} ExtendedServerResponse
@@ -55,6 +56,23 @@ const noop = () => {};
5556
* @param {Stats | MultiStats} [stats]
5657
*/
5758

59+
/**
60+
* @typedef {Object} ResponseData
61+
* @property {string | Buffer | ReadStream} data
62+
* @property {number} byteLength
63+
*/
64+
65+
/**
66+
* @template {IncomingMessage} RequestInternal
67+
* @template {ServerResponse} ResponseInternal
68+
* @callback ModifyResponseData
69+
* @param {RequestInternal} req
70+
* @param {ResponseInternal} res
71+
* @param {string | Buffer | ReadStream} data
72+
* @param {number} byteLength
73+
* @return {ResponseData}
74+
*/
75+
5876
/**
5977
* @template {IncomingMessage} RequestInternal
6078
* @template {ServerResponse} ResponseInternal
@@ -89,6 +107,7 @@ const noop = () => {};
89107
* @property {boolean} [serverSideRender]
90108
* @property {OutputFileSystem} [outputFileSystem]
91109
* @property {boolean | string} [index]
110+
* @property {ModifyResponseData<RequestInternal, ResponseInternal>} [modifyResponseData]
92111
*/
93112

94113
/**

src/middleware.js

+28-7
Original file line numberDiff line numberDiff line change
@@ -202,15 +202,26 @@ function wrapper(context) {
202202
);
203203
setHeaderForResponse(res, "Content-Type", "text/html; charset=utf-8");
204204

205-
const document = createHtmlDocument(416, `Error: ${message}`);
206-
const byteLength = Buffer.byteLength(document);
205+
/** @type {string | Buffer | import("fs").ReadStream} */
206+
let document = createHtmlDocument(416, `Error: ${message}`);
207+
let byteLength = Buffer.byteLength(document);
207208

208209
setHeaderForResponse(
209210
res,
210211
"Content-Length",
211212
Buffer.byteLength(document)
212213
);
213214

215+
if (context.options.modifyResponseData) {
216+
({ data: document, byteLength } =
217+
context.options.modifyResponseData(
218+
req,
219+
res,
220+
document,
221+
byteLength
222+
));
223+
}
224+
214225
send(req, res, document, byteLength);
215226

216227
return;
@@ -244,7 +255,7 @@ function wrapper(context) {
244255
const isFsSupportsStream =
245256
typeof context.outputFileSystem.createReadStream === "function";
246257

247-
let bufferOtStream;
258+
let bufferOrStream;
248259
let byteLength;
249260

250261
try {
@@ -253,26 +264,36 @@ function wrapper(context) {
253264
typeof end !== "undefined" &&
254265
isFsSupportsStream
255266
) {
256-
bufferOtStream =
267+
bufferOrStream =
257268
/** @type {import("fs").createReadStream} */
258269
(context.outputFileSystem.createReadStream)(filename, {
259270
start,
260271
end,
261272
});
262273
byteLength = end - start + 1;
263274
} else {
264-
bufferOtStream = /** @type {import("fs").readFileSync} */ (
275+
bufferOrStream = /** @type {import("fs").readFileSync} */ (
265276
context.outputFileSystem.readFileSync
266277
)(filename);
267-
({ byteLength } = bufferOtStream);
278+
({ byteLength } = bufferOrStream);
268279
}
269280
} catch (_ignoreError) {
270281
await goNext();
271282

272283
return;
273284
}
274285

275-
send(req, res, bufferOtStream, byteLength);
286+
if (context.options.modifyResponseData) {
287+
({ data: bufferOrStream, byteLength } =
288+
context.options.modifyResponseData(
289+
req,
290+
res,
291+
bufferOrStream,
292+
byteLength
293+
));
294+
}
295+
296+
send(req, res, bufferOrStream, byteLength);
276297
}
277298
};
278299
}

src/options.json

+5
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,11 @@
124124
"minLength": 1
125125
}
126126
]
127+
},
128+
"modifyResponseData": {
129+
"description": "Allows to set up a callback to change the response data.",
130+
"link": "https://github.com/webpack/webpack-dev-middleware#modifyresponsedata",
131+
"instanceof": "Function"
127132
}
128133
},
129134
"additionalProperties": false

test/__snapshots__/validation-options.test.js.snap.webpack5

+14
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,13 @@ exports[`validation should throw an error on the "methods" option with "true" va
7777
-> Read more at https://github.com/webpack/webpack-dev-middleware#methods"
7878
`;
7979

80+
exports[`validation should throw an error on the "mimeTypeDefault" option with "0" value 1`] = `
81+
"Invalid options object. Dev Middleware has been initialized using an options object that does not match the API schema.
82+
- options.mimeTypeDefault should be a string.
83+
-> Allows a user to register a default mime type when we can't determine the content type.
84+
-> Read more at https://github.com/webpack/webpack-dev-middleware#mimetypedefault"
85+
`;
86+
8087
exports[`validation should throw an error on the "mimeTypes" option with "foo" value 1`] = `
8188
"Invalid options object. Dev Middleware has been initialized using an options object that does not match the API schema.
8289
- options.mimeTypes should be an object:
@@ -85,6 +92,13 @@ exports[`validation should throw an error on the "mimeTypes" option with "foo" v
8592
-> Read more at https://github.com/webpack/webpack-dev-middleware#mimetypes"
8693
`;
8794

95+
exports[`validation should throw an error on the "modifyResponseData" option with "true" value 1`] = `
96+
"Invalid options object. Dev Middleware has been initialized using an options object that does not match the API schema.
97+
- options.modifyResponseData should be an instance of function.
98+
-> Allows to set up a callback to change the response data.
99+
-> Read more at https://github.com/webpack/webpack-dev-middleware#modifyresponsedata"
100+
`;
101+
88102
exports[`validation should throw an error on the "outputFileSystem" option with "false" value 1`] = `
89103
"Invalid options object. Dev Middleware has been initialized using an options object that does not match the API schema.
90104
- options.outputFileSystem should be an object:

test/helpers/isWebpack5.js

-3
This file was deleted.

0 commit comments

Comments
 (0)