diff --git a/.eslintrc.js b/.eslintrc.js index 654f4758f1..24127cd2b2 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -21,6 +21,21 @@ module.exports = { "prefer-rest-params": "off", strict: ["error", "safe"], "global-require": "off", + "spaced-comment": [ + "error", + "always", + { + line: { + exceptions: ["-", "+"], + markers: ["=", "!", "/"], + }, + block: { + exceptions: ["-", "+"], + markers: ["=", "!"], + balanced: false, + }, + }, + ], }, overrides: [ { diff --git a/CHANGELOG.md b/CHANGELOG.md index 27518517f7..a210e0c039 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,13 @@ All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. +### [4.7.1](https://github.com/webpack/webpack-dev-server/compare/v4.7.0...v4.7.1) (2021-12-22) + + +### Bug Fixes + +* removed `url` package, fixed compatibility with future webpack defaults ([#4132](https://github.com/webpack/webpack-dev-server/issues/4132)) ([4e5d8ea](https://github.com/webpack/webpack-dev-server/commit/4e5d8eae654ef382697722c6406dbc96207594aa)) + ## [4.7.0](https://github.com/webpack/webpack-dev-server/compare/v4.6.0...v4.7.0) (2021-12-21) diff --git a/client-src/clients/SockJSClient.js b/client-src/clients/SockJSClient.js index 0f90bb6da1..2958d44eef 100644 --- a/client-src/clients/SockJSClient.js +++ b/client-src/clients/SockJSClient.js @@ -38,8 +38,12 @@ export default class SockJSClient { * @param {(...args: any[]) => void} f */ onMessage(f) { - this.sock.onmessage = (e) => { - f(e.data); - }; + this.sock.onmessage = + /** + * @param {Error & { data: string }} e + */ + (e) => { + f(e.data); + }; } } diff --git a/client-src/index.js b/client-src/index.js index d8f832c987..dc85fdcff0 100644 --- a/client-src/index.js +++ b/client-src/index.js @@ -1,5 +1,5 @@ /* global __resourceQuery, __webpack_hash__ */ - +/// import webpackHotLog from "webpack/hot/log.js"; import stripAnsi from "./modules/strip-ansi/index.js"; import parseURL from "./utils/parseURL.js"; @@ -10,6 +10,26 @@ import sendMessage from "./utils/sendMessage.js"; import reloadApp from "./utils/reloadApp.js"; import createSocketURL from "./utils/createSocketURL.js"; +/** + * @typedef {Object} Options + * @property {boolean} hot + * @property {boolean} liveReload + * @property {boolean} progress + * @property {boolean | { warnings?: boolean, errors?: boolean }} overlay + * @property {string} [logging] + * @property {number} [reconnect] + */ + +/** + * @typedef {Object} Status + * @property {boolean} isUnloading + * @property {string} currentHash + * @property {string} [previousHash] + */ + +/** + * @type {Status} + */ const status = { isUnloading: false, // TODO Workaround for webpack v4, `__webpack_hash__` is not replaced without HotModuleReplacement @@ -17,6 +37,7 @@ const status = { currentHash: typeof __webpack_hash__ !== "undefined" ? __webpack_hash__ : "", }; +/** @type {Options} */ const options = { hot: false, liveReload: false, @@ -101,6 +122,9 @@ const onSocketMessage = { status.currentHash = hash; }, logging: setAllLogLevel, + /** + * @param {boolean} value + */ overlay(value) { if (typeof document === "undefined") { return; @@ -108,6 +132,9 @@ const onSocketMessage = { options.overlay = value; }, + /** + * @param {number} value + */ reconnect(value) { if (parsedResourceQuery.reconnect === "false") { return; @@ -115,9 +142,15 @@ const onSocketMessage = { options.reconnect = value; }, - progress(progress) { - options.progress = progress; + /** + * @param {boolean} value + */ + progress(value) { + options.progress = value; }, + /** + * @param {{ pluginName?: string, percent: number, msg: string }} data + */ "progress-update": function progressUpdate(data) { if (options.progress) { log.info( @@ -148,6 +181,9 @@ const onSocketMessage = { reloadApp(options, status); }, // TODO: remove in v5 in favor of 'static-changed' + /** + * @param {string} file + */ "content-changed": function contentChanged(file) { log.info( `${ @@ -157,6 +193,9 @@ const onSocketMessage = { self.location.reload(); }, + /** + * @param {string} file + */ "static-changed": function staticChanged(file) { log.info( `${ diff --git a/client-src/overlay.js b/client-src/overlay.js index 819597ed15..5c074702ff 100644 --- a/client-src/overlay.js +++ b/client-src/overlay.js @@ -17,8 +17,11 @@ const colors = { darkgrey: "6D7891", }; +/** @type {HTMLIFrameElement | null | undefined} */ let iframeContainerElement; +/** @type {HTMLDivElement | null | undefined} */ let containerElement; +/** @type {Array<(element: HTMLDivElement) => void>} */ let onLoadQueue = []; ansiHTML.setColors(colors); @@ -38,7 +41,11 @@ function createContainer() { iframeContainerElement.style.zIndex = 9999999999; iframeContainerElement.onload = () => { containerElement = - iframeContainerElement.contentDocument.createElement("div"); + /** @type {Document} */ + ( + /** @type {HTMLIFrameElement} */ + (iframeContainerElement).contentDocument + ).createElement("div"); containerElement.id = "webpack-dev-server-client-overlay-div"; containerElement.style.position = "fixed"; containerElement.style.boxSizing = "border-box"; @@ -71,6 +78,7 @@ function createContainer() { closeButtonElement.style.color = "white"; closeButtonElement.style.cursor = "pointer"; closeButtonElement.style.cssFloat = "right"; + // @ts-ignore closeButtonElement.style.styleFloat = "right"; closeButtonElement.addEventListener("click", () => { hide(); @@ -81,19 +89,27 @@ function createContainer() { containerElement.appendChild(document.createElement("br")); containerElement.appendChild(document.createElement("br")); - iframeContainerElement.contentDocument.body.appendChild(containerElement); + /** @type {Document} */ + ( + /** @type {HTMLIFrameElement} */ + (iframeContainerElement).contentDocument + ).body.appendChild(containerElement); onLoadQueue.forEach((onLoad) => { - onLoad(containerElement); + onLoad(/** @type {HTMLDivElement} */ (containerElement)); }); onLoadQueue = []; - iframeContainerElement.onload = null; + /** @type {HTMLIFrameElement} */ + (iframeContainerElement).onload = null; }; document.body.appendChild(iframeContainerElement); } +/** + * @param {(element: HTMLDivElement) => void} callback + */ function ensureOverlayExists(callback) { if (containerElement) { // Everything is ready, call the callback right away. @@ -124,6 +140,11 @@ function hide() { containerElement = null; } +/** + * @param {string} type + * @param {string | { file?: string, moduleName?: string, loc?: string, message?: string }} item + * @returns {{ header: string, body: string }} + */ function formatProblem(type, item) { let header = type === "warning" ? "WARNING" : "ERROR"; let body = ""; @@ -154,6 +175,10 @@ function formatProblem(type, item) { } // Compilation with errors (e.g. syntax error or missing modules). +/** + * @param {string} type + * @param {Array} messages + */ function show(type, messages) { ensureOverlayExists(() => { messages.forEach((message) => { @@ -177,7 +202,8 @@ function show(type, messages) { entryElement.appendChild(document.createElement("br")); entryElement.appendChild(document.createElement("br")); - containerElement.appendChild(entryElement); + /** @type {HTMLDivElement} */ + (containerElement).appendChild(entryElement); }); }); } diff --git a/client-src/socket.js b/client-src/socket.js index dc76cd295b..4b913e4c69 100644 --- a/client-src/socket.js +++ b/client-src/socket.js @@ -18,12 +18,20 @@ let retries = 0; let maxRetries = 10; let client = null; +/** + * @param {string} url + * @param {{ [handler: string]: (data?: any, params?: any) => any }} handlers + * @param {number} [reconnect] + */ const socket = function initSocket(url, handlers, reconnect) { client = new Client(url); client.onOpen(() => { retries = 0; - maxRetries = reconnect; + + if (typeof reconnect !== "undefined") { + maxRetries = reconnect; + } }); client.onClose(() => { @@ -51,13 +59,18 @@ const socket = function initSocket(url, handlers, reconnect) { } }); - client.onMessage((data) => { - const message = JSON.parse(data); + client.onMessage( + /** + * @param {any} data + */ + (data) => { + const message = JSON.parse(data); - if (handlers[message.type]) { - handlers[message.type](message.data, message.params); + if (handlers[message.type]) { + handlers[message.type](message.data, message.params); + } } - }); + ); }; export default socket; diff --git a/client-src/utils/createSocketURL.js b/client-src/utils/createSocketURL.js index 168c3b6c9e..5d8a28ddbd 100644 --- a/client-src/utils/createSocketURL.js +++ b/client-src/utils/createSocketURL.js @@ -1,12 +1,82 @@ -import url from "url"; +/** + * @param {{ protocol?: string, auth?: string, hostname?: string, port?: string, pathname?: string, search?: string, hash?: string, slashes?: boolean }} objURL + * @returns {string} + */ +function format(objURL) { + let protocol = objURL.protocol || ""; + + if (protocol && protocol.substr(-1) !== ":") { + protocol += ":"; + } + + let auth = objURL.auth || ""; + + if (auth) { + auth = encodeURIComponent(auth); + auth = auth.replace(/%3A/i, ":"); + auth += "@"; + } + + let host = ""; + + if (objURL.hostname) { + host = + auth + + (objURL.hostname.indexOf(":") === -1 + ? objURL.hostname + : `[${objURL.hostname}]`); + + if (objURL.port) { + host += `:${objURL.port}`; + } + } + + let pathname = objURL.pathname || ""; + + if (objURL.slashes) { + host = `//${host || ""}`; + + if (pathname && pathname.charAt(0) !== "/") { + pathname = `/${pathname}`; + } + } else if (!host) { + host = ""; + } + + let search = objURL.search || ""; + + if (search && search.charAt(0) !== "?") { + search = `?${search}`; + } + + let hash = objURL.hash || ""; + + if (hash && hash.charAt(0) !== "#") { + hash = `#${hash}`; + } + + pathname = pathname.replace( + /[?#]/g, + /** + * @param {string} match + * @returns {string} + */ + (match) => encodeURIComponent(match) + ); + search = search.replace("#", "%23"); + + return `${protocol}${host}${pathname}${search}${hash}`; +} -// We handle legacy API that is Node.js specific, and a newer API that implements the same WHATWG URL Standard used by web browsers -// Please look at https://nodejs.org/api/url.html#url_url_strings_and_url_objects +/** + * @param {URL & { fromCurrentScript?: boolean }} parsedURL + * @returns {string} + */ function createSocketURL(parsedURL) { let { hostname } = parsedURL; // Node.js module parses it as `::` - // `new URL(urlString, [baseURLstring])` parses it as '[::]' + // `new URL(urlString, [baseURLString])` parses it as '[::]' const isInAddrAny = hostname === "0.0.0.0" || hostname === "::" || hostname === "[::]"; @@ -80,7 +150,7 @@ function createSocketURL(parsedURL) { socketURLPathname = parsedURL.pathname; } - return url.format({ + return format({ protocol: socketURLProtocol, auth: socketURLAuth, hostname: socketURLHostname, diff --git a/client-src/utils/getCurrentScriptSource.js b/client-src/utils/getCurrentScriptSource.js index 01b05cb66d..2c16610389 100644 --- a/client-src/utils/getCurrentScriptSource.js +++ b/client-src/utils/getCurrentScriptSource.js @@ -1,3 +1,6 @@ +/** + * @returns {string} + */ function getCurrentScriptSource() { // `document.currentScript` is the most accurate way to find the current script, // but is not supported in all browsers. diff --git a/client-src/utils/parseURL.js b/client-src/utils/parseURL.js index 9be4077d25..3a4f2f3fda 100644 --- a/client-src/utils/parseURL.js +++ b/client-src/utils/parseURL.js @@ -1,7 +1,11 @@ -import url from "url"; import getCurrentScriptSource from "./getCurrentScriptSource.js"; +/** + * @param {string} resourceQuery + * @returns {{ [key: string]: string | boolean }} + */ function parseURL(resourceQuery) { + /** @type {{ [key: string]: string }} */ let options = {}; if (typeof resourceQuery === "string" && resourceQuery !== "") { @@ -16,25 +20,20 @@ function parseURL(resourceQuery) { // Else, get the url from the