12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443 |
- "use strict";
- const os = require("os");
- const path = require("path");
- const url = require("url");
- const util = require("util");
- const fs = require("graceful-fs");
- const ipaddr = require("ipaddr.js");
- const defaultGateway = require("default-gateway");
- const express = require("express");
- const { validate } = require("schema-utils");
- const schema = require("./options.json");
- /** @typedef {import("schema-utils/declarations/validate").Schema} Schema */
- /** @typedef {import("webpack").Compiler} Compiler */
- /** @typedef {import("webpack").MultiCompiler} MultiCompiler */
- /** @typedef {import("webpack").Configuration} WebpackConfiguration */
- /** @typedef {import("webpack").StatsOptions} StatsOptions */
- /** @typedef {import("webpack").StatsCompilation} StatsCompilation */
- /** @typedef {import("webpack").Stats} Stats */
- /** @typedef {import("webpack").MultiStats} MultiStats */
- /** @typedef {import("os").NetworkInterfaceInfo} NetworkInterfaceInfo */
- /** @typedef {import("express").Request} Request */
- /** @typedef {import("express").Response} Response */
- /** @typedef {import("express").NextFunction} NextFunction */
- /** @typedef {import("express").RequestHandler} ExpressRequestHandler */
- /** @typedef {import("express").ErrorRequestHandler} ExpressErrorRequestHandler */
- /** @typedef {import("chokidar").WatchOptions} WatchOptions */
- /** @typedef {import("chokidar").FSWatcher} FSWatcher */
- /** @typedef {import("connect-history-api-fallback").Options} ConnectHistoryApiFallbackOptions */
- /** @typedef {import("bonjour-service").Bonjour} Bonjour */
- /** @typedef {import("bonjour-service").Service} BonjourOptions */
- /** @typedef {import("http-proxy-middleware").RequestHandler} RequestHandler */
- /** @typedef {import("http-proxy-middleware").Options} HttpProxyMiddlewareOptions */
- /** @typedef {import("http-proxy-middleware").Filter} HttpProxyMiddlewareOptionsFilter */
- /** @typedef {import("serve-index").Options} ServeIndexOptions */
- /** @typedef {import("serve-static").ServeStaticOptions} ServeStaticOptions */
- /** @typedef {import("ipaddr.js").IPv4} IPv4 */
- /** @typedef {import("ipaddr.js").IPv6} IPv6 */
- /** @typedef {import("net").Socket} Socket */
- /** @typedef {import("http").IncomingMessage} IncomingMessage */
- /** @typedef {import("open").Options} OpenOptions */
- /** @typedef {import("https").ServerOptions & { spdy?: { plain?: boolean | undefined, ssl?: boolean | undefined, 'x-forwarded-for'?: string | undefined, protocol?: string | undefined, protocols?: string[] | undefined }}} ServerOptions */
- /**
- * @template Request, Response
- * @typedef {import("webpack-dev-middleware").Options<Request, Response>} DevMiddlewareOptions
- */
- /**
- * @template Request, Response
- * @typedef {import("webpack-dev-middleware").Context<Request, Response>} DevMiddlewareContext
- */
- /**
- * @typedef {"local-ip" | "local-ipv4" | "local-ipv6" | string} Host
- */
- /**
- * @typedef {number | string | "auto"} Port
- */
- /**
- * @typedef {Object} WatchFiles
- * @property {string | string[]} paths
- * @property {WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean }} [options]
- */
- /**
- * @typedef {Object} Static
- * @property {string} [directory]
- * @property {string | string[]} [publicPath]
- * @property {boolean | ServeIndexOptions} [serveIndex]
- * @property {ServeStaticOptions} [staticOptions]
- * @property {boolean | WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean }} [watch]
- */
- /**
- * @typedef {Object} NormalizedStatic
- * @property {string} directory
- * @property {string[]} publicPath
- * @property {false | ServeIndexOptions} serveIndex
- * @property {ServeStaticOptions} staticOptions
- * @property {false | WatchOptions} watch
- */
- /**
- * @typedef {Object} ServerConfiguration
- * @property {"http" | "https" | "spdy" | string} [type]
- * @property {ServerOptions} [options]
- */
- /**
- * @typedef {Object} WebSocketServerConfiguration
- * @property {"sockjs" | "ws" | string | Function} [type]
- * @property {Record<string, any>} [options]
- */
- /**
- * @typedef {(import("ws").WebSocket | import("sockjs").Connection & { send: import("ws").WebSocket["send"], terminate: import("ws").WebSocket["terminate"], ping: import("ws").WebSocket["ping"] }) & { isAlive?: boolean }} ClientConnection
- */
- /**
- * @typedef {import("ws").WebSocketServer | import("sockjs").Server & { close: import("ws").WebSocketServer["close"] }} WebSocketServer
- */
- /**
- * @typedef {{ implementation: WebSocketServer, clients: ClientConnection[] }} WebSocketServerImplementation
- */
- /**
- * @callback ByPass
- * @param {Request} req
- * @param {Response} res
- * @param {ProxyConfigArrayItem} proxyConfig
- */
- /**
- * @typedef {{ path?: HttpProxyMiddlewareOptionsFilter | undefined, context?: HttpProxyMiddlewareOptionsFilter | undefined } & { bypass?: ByPass } & HttpProxyMiddlewareOptions } ProxyConfigArrayItem
- */
- /**
- * @typedef {(ProxyConfigArrayItem | ((req?: Request | undefined, res?: Response | undefined, next?: NextFunction | undefined) => ProxyConfigArrayItem))[]} ProxyConfigArray
- */
- /**
- * @typedef {{ [url: string]: string | ProxyConfigArrayItem }} ProxyConfigMap
- */
- /**
- * @typedef {Object} OpenApp
- * @property {string} [name]
- * @property {string[]} [arguments]
- */
- /**
- * @typedef {Object} Open
- * @property {string | string[] | OpenApp} [app]
- * @property {string | string[]} [target]
- */
- /**
- * @typedef {Object} NormalizedOpen
- * @property {string} target
- * @property {import("open").Options} options
- */
- /**
- * @typedef {Object} WebSocketURL
- * @property {string} [hostname]
- * @property {string} [password]
- * @property {string} [pathname]
- * @property {number | string} [port]
- * @property {string} [protocol]
- * @property {string} [username]
- */
- /**
- * @typedef {Object} ClientConfiguration
- * @property {"log" | "info" | "warn" | "error" | "none" | "verbose"} [logging]
- * @property {boolean | { warnings?: boolean, errors?: boolean }} [overlay]
- * @property {boolean} [progress]
- * @property {boolean | number} [reconnect]
- * @property {"ws" | "sockjs" | string} [webSocketTransport]
- * @property {string | WebSocketURL} [webSocketURL]
- */
- /**
- * @typedef {Array<{ key: string; value: string }> | Record<string, string | string[]>} Headers
- */
- /**
- * @typedef {{ name?: string, path?: string, middleware: ExpressRequestHandler | ExpressErrorRequestHandler } | ExpressRequestHandler | ExpressErrorRequestHandler} Middleware
- */
- /**
- * @typedef {Object} Configuration
- * @property {boolean | string} [ipc]
- * @property {Host} [host]
- * @property {Port} [port]
- * @property {boolean | "only"} [hot]
- * @property {boolean} [liveReload]
- * @property {DevMiddlewareOptions<Request, Response>} [devMiddleware]
- * @property {boolean} [compress]
- * @property {boolean} [magicHtml]
- * @property {"auto" | "all" | string | string[]} [allowedHosts]
- * @property {boolean | ConnectHistoryApiFallbackOptions} [historyApiFallback]
- * @property {boolean} [setupExitSignals]
- * @property {boolean | Record<string, never> | BonjourOptions} [bonjour]
- * @property {string | string[] | WatchFiles | Array<string | WatchFiles>} [watchFiles]
- * @property {boolean | string | Static | Array<string | Static>} [static]
- * @property {boolean | ServerOptions} [https]
- * @property {boolean} [http2]
- * @property {"http" | "https" | "spdy" | string | ServerConfiguration} [server]
- * @property {boolean | "sockjs" | "ws" | string | WebSocketServerConfiguration} [webSocketServer]
- * @property {ProxyConfigMap | ProxyConfigArrayItem | ProxyConfigArray} [proxy]
- * @property {boolean | string | Open | Array<string | Open>} [open]
- * @property {boolean} [setupExitSignals]
- * @property {boolean | ClientConfiguration} [client]
- * @property {Headers | ((req: Request, res: Response, context: DevMiddlewareContext<Request, Response>) => Headers)} [headers]
- * @property {(devServer: Server) => void} [onAfterSetupMiddleware]
- * @property {(devServer: Server) => void} [onBeforeSetupMiddleware]
- * @property {(devServer: Server) => void} [onListening]
- * @property {(middlewares: Middleware[], devServer: Server) => Middleware[]} [setupMiddlewares]
- */
- if (!process.env.WEBPACK_SERVE) {
- // TODO fix me in the next major release
- // @ts-ignore
- process.env.WEBPACK_SERVE = true;
- }
- class Server {
- /**
- * @param {Configuration | Compiler | MultiCompiler} options
- * @param {Compiler | MultiCompiler | Configuration} compiler
- */
- constructor(options = {}, compiler) {
- // TODO: remove this after plugin support is published
- if (/** @type {Compiler | MultiCompiler} */ (options).hooks) {
- util.deprecate(
- () => {},
- "Using 'compiler' as the first argument is deprecated. Please use 'options' as the first argument and 'compiler' as the second argument.",
- "DEP_WEBPACK_DEV_SERVER_CONSTRUCTOR"
- )();
- [options = {}, compiler] = [compiler, options];
- }
- validate(/** @type {Schema} */ (schema), options, {
- name: "Dev Server",
- baseDataPath: "options",
- });
- this.compiler = /** @type {Compiler | MultiCompiler} */ (compiler);
- /**
- * @type {ReturnType<Compiler["getInfrastructureLogger"]>}
- * */
- this.logger = this.compiler.getInfrastructureLogger("webpack-dev-server");
- this.options = /** @type {Configuration} */ (options);
- /**
- * @type {FSWatcher[]}
- */
- this.staticWatchers = [];
- /**
- * @private
- * @type {{ name: string | symbol, listener: (...args: any[]) => void}[] }}
- */
- this.listeners = [];
- // Keep track of websocket proxies for external websocket upgrade.
- /**
- * @private
- * @type {RequestHandler[]}
- */
- this.webSocketProxies = [];
- /**
- * @type {Socket[]}
- */
- this.sockets = [];
- /**
- * @private
- * @type {string | undefined}
- */
- // eslint-disable-next-line no-undefined
- this.currentHash = undefined;
- }
- // TODO compatibility with webpack v4, remove it after drop
- static get cli() {
- return {
- get getArguments() {
- return () => require("../bin/cli-flags");
- },
- get processArguments() {
- return require("../bin/process-arguments");
- },
- };
- }
- static get schema() {
- return schema;
- }
- /**
- * @private
- * @returns {StatsOptions}
- * @constructor
- */
- static get DEFAULT_STATS() {
- return {
- all: false,
- hash: true,
- warnings: true,
- errors: true,
- errorDetails: false,
- };
- }
- /**
- * @param {string} URL
- * @returns {boolean}
- */
- static isAbsoluteURL(URL) {
- // Don't match Windows paths `c:\`
- if (/^[a-zA-Z]:\\/.test(URL)) {
- return false;
- }
- // Scheme: https://tools.ietf.org/html/rfc3986#section-3.1
- // Absolute URL: https://tools.ietf.org/html/rfc3986#section-4.3
- return /^[a-zA-Z][a-zA-Z\d+\-.]*:/.test(URL);
- }
- /**
- * @param {string} gateway
- * @returns {string | undefined}
- */
- static findIp(gateway) {
- const gatewayIp = ipaddr.parse(gateway);
- // Look for the matching interface in all local interfaces.
- for (const addresses of Object.values(os.networkInterfaces())) {
- for (const { cidr } of /** @type {NetworkInterfaceInfo[]} */ (
- addresses
- )) {
- const net = ipaddr.parseCIDR(/** @type {string} */ (cidr));
- if (
- net[0] &&
- net[0].kind() === gatewayIp.kind() &&
- gatewayIp.match(net)
- ) {
- return net[0].toString();
- }
- }
- }
- }
- /**
- * @param {"v4" | "v6"} family
- * @returns {Promise<string | undefined>}
- */
- static async internalIP(family) {
- try {
- const { gateway } = await defaultGateway[family]();
- return Server.findIp(gateway);
- } catch {
- // ignore
- }
- }
- /**
- * @param {"v4" | "v6"} family
- * @returns {string | undefined}
- */
- static internalIPSync(family) {
- try {
- const { gateway } = defaultGateway[family].sync();
- return Server.findIp(gateway);
- } catch {
- // ignore
- }
- }
- /**
- * @param {Host} hostname
- * @returns {Promise<string>}
- */
- static async getHostname(hostname) {
- if (hostname === "local-ip") {
- return (
- (await Server.internalIP("v4")) ||
- (await Server.internalIP("v6")) ||
- "0.0.0.0"
- );
- } else if (hostname === "local-ipv4") {
- return (await Server.internalIP("v4")) || "0.0.0.0";
- } else if (hostname === "local-ipv6") {
- return (await Server.internalIP("v6")) || "::";
- }
- return hostname;
- }
- /**
- * @param {Port} port
- * @returns {Promise<number | string>}
- */
- static async getFreePort(port) {
- if (typeof port !== "undefined" && port !== null && port !== "auto") {
- return port;
- }
- const pRetry = require("p-retry");
- const portfinder = require("portfinder");
- portfinder.basePort =
- typeof process.env.WEBPACK_DEV_SERVER_BASE_PORT !== "undefined"
- ? parseInt(process.env.WEBPACK_DEV_SERVER_BASE_PORT, 10)
- : 8080;
- // Try to find unused port and listen on it for 3 times,
- // if port is not specified in options.
- const defaultPortRetry =
- typeof process.env.WEBPACK_DEV_SERVER_PORT_RETRY !== "undefined"
- ? parseInt(process.env.WEBPACK_DEV_SERVER_PORT_RETRY, 10)
- : 3;
- return pRetry(() => portfinder.getPortPromise(), {
- retries: defaultPortRetry,
- });
- }
- /**
- * @returns {string}
- */
- static findCacheDir() {
- const cwd = process.cwd();
- /**
- * @type {string | undefined}
- */
- let dir = cwd;
- for (;;) {
- try {
- if (fs.statSync(path.join(dir, "package.json")).isFile()) break;
- // eslint-disable-next-line no-empty
- } catch (e) {}
- const parent = path.dirname(dir);
- if (dir === parent) {
- // eslint-disable-next-line no-undefined
- dir = undefined;
- break;
- }
- dir = parent;
- }
- if (!dir) {
- return path.resolve(cwd, ".cache/webpack-dev-server");
- } else if (process.versions.pnp === "1") {
- return path.resolve(dir, ".pnp/.cache/webpack-dev-server");
- } else if (process.versions.pnp === "3") {
- return path.resolve(dir, ".yarn/.cache/webpack-dev-server");
- }
- return path.resolve(dir, "node_modules/.cache/webpack-dev-server");
- }
- /**
- * @private
- * @param {Compiler} compiler
- */
- addAdditionalEntries(compiler) {
- /**
- * @type {string[]}
- */
- const additionalEntries = [];
- const isWebTarget = compiler.options.externalsPresets
- ? compiler.options.externalsPresets.web
- : [
- "web",
- "webworker",
- "electron-preload",
- "electron-renderer",
- "node-webkit",
- // eslint-disable-next-line no-undefined
- undefined,
- null,
- ].includes(/** @type {string} */ (compiler.options.target));
- // TODO maybe empty empty client
- if (this.options.client && isWebTarget) {
- let webSocketURLStr = "";
- if (this.options.webSocketServer) {
- const webSocketURL =
- /** @type {WebSocketURL} */
- (
- /** @type {ClientConfiguration} */
- (this.options.client).webSocketURL
- );
- const webSocketServer =
- /** @type {{ type: WebSocketServerConfiguration["type"], options: NonNullable<WebSocketServerConfiguration["options"]> }} */
- (this.options.webSocketServer);
- const searchParams = new URLSearchParams();
- /** @type {string} */
- let protocol;
- // We are proxying dev server and need to specify custom `hostname`
- if (typeof webSocketURL.protocol !== "undefined") {
- protocol = webSocketURL.protocol;
- } else {
- protocol =
- /** @type {ServerConfiguration} */
- (this.options.server).type === "http" ? "ws:" : "wss:";
- }
- searchParams.set("protocol", protocol);
- if (typeof webSocketURL.username !== "undefined") {
- searchParams.set("username", webSocketURL.username);
- }
- if (typeof webSocketURL.password !== "undefined") {
- searchParams.set("password", webSocketURL.password);
- }
- /** @type {string} */
- let hostname;
- // SockJS is not supported server mode, so `hostname` and `port` can't specified, let's ignore them
- // TODO show warning about this
- const isSockJSType = webSocketServer.type === "sockjs";
- // We are proxying dev server and need to specify custom `hostname`
- if (typeof webSocketURL.hostname !== "undefined") {
- hostname = webSocketURL.hostname;
- }
- // Web socket server works on custom `hostname`, only for `ws` because `sock-js` is not support custom `hostname`
- else if (
- typeof webSocketServer.options.host !== "undefined" &&
- !isSockJSType
- ) {
- hostname = webSocketServer.options.host;
- }
- // The `host` option is specified
- else if (typeof this.options.host !== "undefined") {
- hostname = this.options.host;
- }
- // The `port` option is not specified
- else {
- hostname = "0.0.0.0";
- }
- searchParams.set("hostname", hostname);
- /** @type {number | string} */
- let port;
- // We are proxying dev server and need to specify custom `port`
- if (typeof webSocketURL.port !== "undefined") {
- port = webSocketURL.port;
- }
- // Web socket server works on custom `port`, only for `ws` because `sock-js` is not support custom `port`
- else if (
- typeof webSocketServer.options.port !== "undefined" &&
- !isSockJSType
- ) {
- port = webSocketServer.options.port;
- }
- // The `port` option is specified
- else if (typeof this.options.port === "number") {
- port = this.options.port;
- }
- // The `port` option is specified using `string`
- else if (
- typeof this.options.port === "string" &&
- this.options.port !== "auto"
- ) {
- port = Number(this.options.port);
- }
- // The `port` option is not specified or set to `auto`
- else {
- port = "0";
- }
- searchParams.set("port", String(port));
- /** @type {string} */
- let pathname = "";
- // We are proxying dev server and need to specify custom `pathname`
- if (typeof webSocketURL.pathname !== "undefined") {
- pathname = webSocketURL.pathname;
- }
- // Web socket server works on custom `path`
- else if (
- typeof webSocketServer.options.prefix !== "undefined" ||
- typeof webSocketServer.options.path !== "undefined"
- ) {
- pathname =
- webSocketServer.options.prefix || webSocketServer.options.path;
- }
- searchParams.set("pathname", pathname);
- const client = /** @type {ClientConfiguration} */ (this.options.client);
- if (typeof client.logging !== "undefined") {
- searchParams.set("logging", client.logging);
- }
- if (typeof client.reconnect !== "undefined") {
- searchParams.set(
- "reconnect",
- typeof client.reconnect === "number"
- ? String(client.reconnect)
- : "10"
- );
- }
- webSocketURLStr = searchParams.toString();
- }
- additionalEntries.push(
- `${require.resolve("../client/index.js")}?${webSocketURLStr}`
- );
- }
- if (this.options.hot === "only") {
- additionalEntries.push(require.resolve("webpack/hot/only-dev-server"));
- } else if (this.options.hot) {
- additionalEntries.push(require.resolve("webpack/hot/dev-server"));
- }
- const webpack = compiler.webpack || require("webpack");
- // use a hook to add entries if available
- if (typeof webpack.EntryPlugin !== "undefined") {
- for (const additionalEntry of additionalEntries) {
- new webpack.EntryPlugin(compiler.context, additionalEntry, {
- // eslint-disable-next-line no-undefined
- name: undefined,
- }).apply(compiler);
- }
- }
- // TODO remove after drop webpack v4 support
- else {
- /**
- * prependEntry Method for webpack 4
- * @param {any} originalEntry
- * @param {any} newAdditionalEntries
- * @returns {any}
- */
- const prependEntry = (originalEntry, newAdditionalEntries) => {
- if (typeof originalEntry === "function") {
- return () =>
- Promise.resolve(originalEntry()).then((entry) =>
- prependEntry(entry, newAdditionalEntries)
- );
- }
- if (
- typeof originalEntry === "object" &&
- !Array.isArray(originalEntry)
- ) {
- /** @type {Object<string,string>} */
- const clone = {};
- Object.keys(originalEntry).forEach((key) => {
- // entry[key] should be a string here
- const entryDescription = originalEntry[key];
- clone[key] = prependEntry(entryDescription, newAdditionalEntries);
- });
- return clone;
- }
- // in this case, entry is a string or an array.
- // make sure that we do not add duplicates.
- /** @type {any} */
- const entriesClone = additionalEntries.slice(0);
- [].concat(originalEntry).forEach((newEntry) => {
- if (!entriesClone.includes(newEntry)) {
- entriesClone.push(newEntry);
- }
- });
- return entriesClone;
- };
- compiler.options.entry = prependEntry(
- compiler.options.entry || "./src",
- additionalEntries
- );
- compiler.hooks.entryOption.call(
- /** @type {string} */ (compiler.options.context),
- compiler.options.entry
- );
- }
- }
- /**
- * @private
- * @returns {Compiler["options"]}
- */
- getCompilerOptions() {
- if (
- typeof (/** @type {MultiCompiler} */ (this.compiler).compilers) !==
- "undefined"
- ) {
- if (/** @type {MultiCompiler} */ (this.compiler).compilers.length === 1) {
- return (
- /** @type {MultiCompiler} */
- (this.compiler).compilers[0].options
- );
- }
- // Configuration with the `devServer` options
- const compilerWithDevServer =
- /** @type {MultiCompiler} */
- (this.compiler).compilers.find((config) => config.options.devServer);
- if (compilerWithDevServer) {
- return compilerWithDevServer.options;
- }
- // Configuration with `web` preset
- const compilerWithWebPreset =
- /** @type {MultiCompiler} */
- (this.compiler).compilers.find(
- (config) =>
- (config.options.externalsPresets &&
- config.options.externalsPresets.web) ||
- [
- "web",
- "webworker",
- "electron-preload",
- "electron-renderer",
- "node-webkit",
- // eslint-disable-next-line no-undefined
- undefined,
- null,
- ].includes(/** @type {string} */ (config.options.target))
- );
- if (compilerWithWebPreset) {
- return compilerWithWebPreset.options;
- }
- // Fallback
- return /** @type {MultiCompiler} */ (this.compiler).compilers[0].options;
- }
- return /** @type {Compiler} */ (this.compiler).options;
- }
- /**
- * @private
- * @returns {Promise<void>}
- */
- async normalizeOptions() {
- const { options } = this;
- const compilerOptions = this.getCompilerOptions();
- // TODO remove `{}` after drop webpack v4 support
- const compilerWatchOptions = compilerOptions.watchOptions || {};
- /**
- * @param {WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean }} watchOptions
- * @returns {WatchOptions}
- */
- const getWatchOptions = (watchOptions = {}) => {
- const getPolling = () => {
- if (typeof watchOptions.usePolling !== "undefined") {
- return watchOptions.usePolling;
- }
- if (typeof watchOptions.poll !== "undefined") {
- return Boolean(watchOptions.poll);
- }
- if (typeof compilerWatchOptions.poll !== "undefined") {
- return Boolean(compilerWatchOptions.poll);
- }
- return false;
- };
- const getInterval = () => {
- if (typeof watchOptions.interval !== "undefined") {
- return watchOptions.interval;
- }
- if (typeof watchOptions.poll === "number") {
- return watchOptions.poll;
- }
- if (typeof compilerWatchOptions.poll === "number") {
- return compilerWatchOptions.poll;
- }
- };
- const usePolling = getPolling();
- const interval = getInterval();
- const { poll, ...rest } = watchOptions;
- return {
- ignoreInitial: true,
- persistent: true,
- followSymlinks: false,
- atomic: false,
- alwaysStat: true,
- ignorePermissionErrors: true,
- // Respect options from compiler watchOptions
- usePolling,
- interval,
- ignored: watchOptions.ignored,
- // TODO: we respect these options for all watch options and allow developers to pass them to chokidar, but chokidar doesn't have these options maybe we need revisit that in future
- ...rest,
- };
- };
- /**
- * @param {string | Static | undefined} [optionsForStatic]
- * @returns {NormalizedStatic}
- */
- const getStaticItem = (optionsForStatic) => {
- const getDefaultStaticOptions = () => {
- return {
- directory: path.join(process.cwd(), "public"),
- staticOptions: {},
- publicPath: ["/"],
- serveIndex: { icons: true },
- watch: getWatchOptions(),
- };
- };
- /** @type {NormalizedStatic} */
- let item;
- if (typeof optionsForStatic === "undefined") {
- item = getDefaultStaticOptions();
- } else if (typeof optionsForStatic === "string") {
- item = {
- ...getDefaultStaticOptions(),
- directory: optionsForStatic,
- };
- } else {
- const def = getDefaultStaticOptions();
- item = {
- directory:
- typeof optionsForStatic.directory !== "undefined"
- ? optionsForStatic.directory
- : def.directory,
- // TODO: do merge in the next major release
- staticOptions:
- typeof optionsForStatic.staticOptions !== "undefined"
- ? optionsForStatic.staticOptions
- : def.staticOptions,
- publicPath:
- // eslint-disable-next-line no-nested-ternary
- typeof optionsForStatic.publicPath !== "undefined"
- ? Array.isArray(optionsForStatic.publicPath)
- ? optionsForStatic.publicPath
- : [optionsForStatic.publicPath]
- : def.publicPath,
- // TODO: do merge in the next major release
- serveIndex:
- // eslint-disable-next-line no-nested-ternary
- typeof optionsForStatic.serveIndex !== "undefined"
- ? typeof optionsForStatic.serveIndex === "boolean" &&
- optionsForStatic.serveIndex
- ? def.serveIndex
- : optionsForStatic.serveIndex
- : def.serveIndex,
- watch:
- // eslint-disable-next-line no-nested-ternary
- typeof optionsForStatic.watch !== "undefined"
- ? // eslint-disable-next-line no-nested-ternary
- typeof optionsForStatic.watch === "boolean"
- ? optionsForStatic.watch
- ? def.watch
- : false
- : getWatchOptions(optionsForStatic.watch)
- : def.watch,
- };
- }
- if (Server.isAbsoluteURL(item.directory)) {
- throw new Error("Using a URL as static.directory is not supported");
- }
- return item;
- };
- if (typeof options.allowedHosts === "undefined") {
- // AllowedHosts allows some default hosts picked from `options.host` or `webSocketURL.hostname` and `localhost`
- options.allowedHosts = "auto";
- }
- // We store allowedHosts as array when supplied as string
- else if (
- typeof options.allowedHosts === "string" &&
- options.allowedHosts !== "auto" &&
- options.allowedHosts !== "all"
- ) {
- options.allowedHosts = [options.allowedHosts];
- }
- // CLI pass options as array, we should normalize them
- else if (
- Array.isArray(options.allowedHosts) &&
- options.allowedHosts.includes("all")
- ) {
- options.allowedHosts = "all";
- }
- if (typeof options.bonjour === "undefined") {
- options.bonjour = false;
- } else if (typeof options.bonjour === "boolean") {
- options.bonjour = options.bonjour ? {} : false;
- }
- if (
- typeof options.client === "undefined" ||
- (typeof options.client === "object" && options.client !== null)
- ) {
- if (!options.client) {
- options.client = {};
- }
- if (typeof options.client.webSocketURL === "undefined") {
- options.client.webSocketURL = {};
- } else if (typeof options.client.webSocketURL === "string") {
- const parsedURL = new URL(options.client.webSocketURL);
- options.client.webSocketURL = {
- protocol: parsedURL.protocol,
- hostname: parsedURL.hostname,
- port: parsedURL.port.length > 0 ? Number(parsedURL.port) : "",
- pathname: parsedURL.pathname,
- username: parsedURL.username,
- password: parsedURL.password,
- };
- } else if (typeof options.client.webSocketURL.port === "string") {
- options.client.webSocketURL.port = Number(
- options.client.webSocketURL.port
- );
- }
- // Enable client overlay by default
- if (typeof options.client.overlay === "undefined") {
- options.client.overlay = true;
- } else if (typeof options.client.overlay !== "boolean") {
- options.client.overlay = {
- errors: true,
- warnings: true,
- ...options.client.overlay,
- };
- }
- if (typeof options.client.reconnect === "undefined") {
- options.client.reconnect = 10;
- } else if (options.client.reconnect === true) {
- options.client.reconnect = Infinity;
- } else if (options.client.reconnect === false) {
- options.client.reconnect = 0;
- }
- // Respect infrastructureLogging.level
- if (typeof options.client.logging === "undefined") {
- options.client.logging = compilerOptions.infrastructureLogging
- ? compilerOptions.infrastructureLogging.level
- : "info";
- }
- }
- if (typeof options.compress === "undefined") {
- options.compress = true;
- }
- if (typeof options.devMiddleware === "undefined") {
- options.devMiddleware = {};
- }
- // No need to normalize `headers`
- if (typeof options.historyApiFallback === "undefined") {
- options.historyApiFallback = false;
- } else if (
- typeof options.historyApiFallback === "boolean" &&
- options.historyApiFallback
- ) {
- options.historyApiFallback = {};
- }
- // No need to normalize `host`
- options.hot =
- typeof options.hot === "boolean" || options.hot === "only"
- ? options.hot
- : true;
- const isHTTPs = Boolean(options.https);
- const isSPDY = Boolean(options.http2);
- if (isHTTPs) {
- // TODO: remove in the next major release
- util.deprecate(
- () => {},
- "'https' option is deprecated. Please use the 'server' option.",
- "DEP_WEBPACK_DEV_SERVER_HTTPS"
- )();
- }
- if (isSPDY) {
- // TODO: remove in the next major release
- util.deprecate(
- () => {},
- "'http2' option is deprecated. Please use the 'server' option.",
- "DEP_WEBPACK_DEV_SERVER_HTTP2"
- )();
- }
- options.server = {
- type:
- // eslint-disable-next-line no-nested-ternary
- typeof options.server === "string"
- ? options.server
- : // eslint-disable-next-line no-nested-ternary
- typeof (options.server || {}).type === "string"
- ? /** @type {ServerConfiguration} */ (options.server).type || "http"
- : // eslint-disable-next-line no-nested-ternary
- isSPDY
- ? "spdy"
- : isHTTPs
- ? "https"
- : "http",
- options: {
- .../** @type {ServerOptions} */ (options.https),
- .../** @type {ServerConfiguration} */ (options.server || {}).options,
- },
- };
- if (
- options.server.type === "spdy" &&
- typeof (/** @type {ServerOptions} */ (options.server.options).spdy) ===
- "undefined"
- ) {
- /** @type {ServerOptions} */
- (options.server.options).spdy = {
- protocols: ["h2", "http/1.1"],
- };
- }
- if (options.server.type === "https" || options.server.type === "spdy") {
- if (
- typeof (
- /** @type {ServerOptions} */ (options.server.options).requestCert
- ) === "undefined"
- ) {
- /** @type {ServerOptions} */
- (options.server.options).requestCert = false;
- }
- const httpsProperties =
- /** @type {Array<keyof ServerOptions>} */
- (["cacert", "ca", "cert", "crl", "key", "pfx"]);
- for (const property of httpsProperties) {
- if (
- typeof (
- /** @type {ServerOptions} */ (options.server.options)[property]
- ) === "undefined"
- ) {
- // eslint-disable-next-line no-continue
- continue;
- }
- // @ts-ignore
- if (property === "cacert") {
- // TODO remove the `cacert` option in favor `ca` in the next major release
- util.deprecate(
- () => {},
- "The 'cacert' option is deprecated. Please use the 'ca' option.",
- "DEP_WEBPACK_DEV_SERVER_CACERT"
- )();
- }
- /** @type {any} */
- const value =
- /** @type {ServerOptions} */
- (options.server.options)[property];
- /**
- * @param {string | Buffer | undefined} item
- * @returns {string | Buffer | undefined}
- */
- const readFile = (item) => {
- if (
- Buffer.isBuffer(item) ||
- (typeof item === "object" && item !== null && !Array.isArray(item))
- ) {
- return item;
- }
- if (item) {
- let stats = null;
- try {
- stats = fs.lstatSync(fs.realpathSync(item)).isFile();
- } catch (error) {
- // Ignore error
- }
- // It is file
- return stats ? fs.readFileSync(item) : item;
- }
- };
- /** @type {any} */
- (options.server.options)[property] = Array.isArray(value)
- ? value.map((item) => readFile(item))
- : readFile(value);
- }
- let fakeCert;
- if (
- !(/** @type {ServerOptions} */ (options.server.options).key) ||
- /** @type {ServerOptions} */ (!options.server.options).cert
- ) {
- const certificateDir = Server.findCacheDir();
- const certificatePath = path.join(certificateDir, "server.pem");
- let certificateExists;
- try {
- const certificate = await fs.promises.stat(certificatePath);
- certificateExists = certificate.isFile();
- } catch {
- certificateExists = false;
- }
- if (certificateExists) {
- const certificateTtl = 1000 * 60 * 60 * 24;
- const certificateStat = await fs.promises.stat(certificatePath);
- const now = Number(new Date());
- // cert is more than 30 days old, kill it with fire
- if ((now - Number(certificateStat.ctime)) / certificateTtl > 30) {
- const { promisify } = require("util");
- const rimraf = require("rimraf");
- const del = promisify(rimraf);
- this.logger.info(
- "SSL certificate is more than 30 days old. Removing..."
- );
- await del(certificatePath);
- certificateExists = false;
- }
- }
- if (!certificateExists) {
- this.logger.info("Generating SSL certificate...");
- // @ts-ignore
- const selfsigned = require("selfsigned");
- const attributes = [{ name: "commonName", value: "localhost" }];
- const pems = selfsigned.generate(attributes, {
- algorithm: "sha256",
- days: 30,
- keySize: 2048,
- extensions: [
- {
- name: "basicConstraints",
- cA: true,
- },
- {
- name: "keyUsage",
- keyCertSign: true,
- digitalSignature: true,
- nonRepudiation: true,
- keyEncipherment: true,
- dataEncipherment: true,
- },
- {
- name: "extKeyUsage",
- serverAuth: true,
- clientAuth: true,
- codeSigning: true,
- timeStamping: true,
- },
- {
- name: "subjectAltName",
- altNames: [
- {
- // type 2 is DNS
- type: 2,
- value: "localhost",
- },
- {
- type: 2,
- value: "localhost.localdomain",
- },
- {
- type: 2,
- value: "lvh.me",
- },
- {
- type: 2,
- value: "*.lvh.me",
- },
- {
- type: 2,
- value: "[::1]",
- },
- {
- // type 7 is IP
- type: 7,
- ip: "127.0.0.1",
- },
- {
- type: 7,
- ip: "fe80::1",
- },
- ],
- },
- ],
- });
- await fs.promises.mkdir(certificateDir, { recursive: true });
- await fs.promises.writeFile(
- certificatePath,
- pems.private + pems.cert,
- {
- encoding: "utf8",
- }
- );
- }
- fakeCert = await fs.promises.readFile(certificatePath);
- this.logger.info(`SSL certificate: ${certificatePath}`);
- }
- if (
- /** @type {ServerOptions & { cacert?: ServerOptions["ca"] }} */ (
- options.server.options
- ).cacert
- ) {
- if (/** @type {ServerOptions} */ (options.server.options).ca) {
- this.logger.warn(
- "Do not specify 'ca' and 'cacert' options together, the 'ca' option will be used."
- );
- } else {
- /** @type {ServerOptions} */
- (options.server.options).ca =
- /** @type {ServerOptions & { cacert?: ServerOptions["ca"] }} */
- (options.server.options).cacert;
- }
- delete (
- /** @type {ServerOptions & { cacert?: ServerOptions["ca"] }} */ (
- options.server.options
- ).cacert
- );
- }
- /** @type {ServerOptions} */
- (options.server.options).key =
- /** @type {ServerOptions} */
- (options.server.options).key || fakeCert;
- /** @type {ServerOptions} */
- (options.server.options).cert =
- /** @type {ServerOptions} */
- (options.server.options).cert || fakeCert;
- }
- if (typeof options.ipc === "boolean") {
- const isWindows = process.platform === "win32";
- const pipePrefix = isWindows ? "\\\\.\\pipe\\" : os.tmpdir();
- const pipeName = "webpack-dev-server.sock";
- options.ipc = path.join(pipePrefix, pipeName);
- }
- options.liveReload =
- typeof options.liveReload !== "undefined" ? options.liveReload : true;
- options.magicHtml =
- typeof options.magicHtml !== "undefined" ? options.magicHtml : true;
- // https://github.com/webpack/webpack-dev-server/issues/1990
- const defaultOpenOptions = { wait: false };
- /**
- * @param {any} target
- * @returns {NormalizedOpen[]}
- */
- // TODO: remove --open-app in favor of --open-app-name
- const getOpenItemsFromObject = ({ target, ...rest }) => {
- const normalizedOptions = { ...defaultOpenOptions, ...rest };
- if (typeof normalizedOptions.app === "string") {
- normalizedOptions.app = {
- name: normalizedOptions.app,
- };
- }
- const normalizedTarget = typeof target === "undefined" ? "<url>" : target;
- if (Array.isArray(normalizedTarget)) {
- return normalizedTarget.map((singleTarget) => {
- return { target: singleTarget, options: normalizedOptions };
- });
- }
- return [{ target: normalizedTarget, options: normalizedOptions }];
- };
- if (typeof options.open === "undefined") {
- /** @type {NormalizedOpen[]} */
- (options.open) = [];
- } else if (typeof options.open === "boolean") {
- /** @type {NormalizedOpen[]} */
- (options.open) = options.open
- ? [
- {
- target: "<url>",
- options: /** @type {OpenOptions} */ (defaultOpenOptions),
- },
- ]
- : [];
- } else if (typeof options.open === "string") {
- /** @type {NormalizedOpen[]} */
- (options.open) = [{ target: options.open, options: defaultOpenOptions }];
- } else if (Array.isArray(options.open)) {
- /**
- * @type {NormalizedOpen[]}
- */
- const result = [];
- options.open.forEach((item) => {
- if (typeof item === "string") {
- result.push({ target: item, options: defaultOpenOptions });
- return;
- }
- result.push(...getOpenItemsFromObject(item));
- });
- /** @type {NormalizedOpen[]} */
- (options.open) = result;
- } else {
- /** @type {NormalizedOpen[]} */
- (options.open) = [...getOpenItemsFromObject(options.open)];
- }
- if (options.onAfterSetupMiddleware) {
- // TODO: remove in the next major release
- util.deprecate(
- () => {},
- "'onAfterSetupMiddleware' option is deprecated. Please use the 'setupMiddlewares' option.",
- `DEP_WEBPACK_DEV_SERVER_ON_AFTER_SETUP_MIDDLEWARE`
- )();
- }
- if (options.onBeforeSetupMiddleware) {
- // TODO: remove in the next major release
- util.deprecate(
- () => {},
- "'onBeforeSetupMiddleware' option is deprecated. Please use the 'setupMiddlewares' option.",
- `DEP_WEBPACK_DEV_SERVER_ON_BEFORE_SETUP_MIDDLEWARE`
- )();
- }
- if (typeof options.port === "string" && options.port !== "auto") {
- options.port = Number(options.port);
- }
- /**
- * Assume a proxy configuration specified as:
- * proxy: {
- * 'context': { options }
- * }
- * OR
- * proxy: {
- * 'context': 'target'
- * }
- */
- if (typeof options.proxy !== "undefined") {
- // TODO remove in the next major release, only accept `Array`
- if (!Array.isArray(options.proxy)) {
- if (
- Object.prototype.hasOwnProperty.call(options.proxy, "target") ||
- Object.prototype.hasOwnProperty.call(options.proxy, "router")
- ) {
- /** @type {ProxyConfigArray} */
- (options.proxy) = [/** @type {ProxyConfigMap} */ (options.proxy)];
- } else {
- /** @type {ProxyConfigArray} */
- (options.proxy) = Object.keys(options.proxy).map(
- /**
- * @param {string} context
- * @returns {HttpProxyMiddlewareOptions}
- */
- (context) => {
- let proxyOptions;
- // For backwards compatibility reasons.
- const correctedContext = context
- .replace(/^\*$/, "**")
- .replace(/\/\*$/, "");
- if (
- typeof (
- /** @type {ProxyConfigMap} */ (options.proxy)[context]
- ) === "string"
- ) {
- proxyOptions = {
- context: correctedContext,
- target:
- /** @type {ProxyConfigMap} */
- (options.proxy)[context],
- };
- } else {
- proxyOptions = {
- // @ts-ignore
- .../** @type {ProxyConfigMap} */ (options.proxy)[context],
- };
- proxyOptions.context = correctedContext;
- }
- return proxyOptions;
- }
- );
- }
- }
- /** @type {ProxyConfigArray} */
- (options.proxy) =
- /** @type {ProxyConfigArray} */
- (options.proxy).map((item) => {
- if (typeof item === "function") {
- return item;
- }
- /**
- * @param {"info" | "warn" | "error" | "debug" | "silent" | undefined | "none" | "log" | "verbose"} level
- * @returns {"info" | "warn" | "error" | "debug" | "silent" | undefined}
- */
- const getLogLevelForProxy = (level) => {
- if (level === "none") {
- return "silent";
- }
- if (level === "log") {
- return "info";
- }
- if (level === "verbose") {
- return "debug";
- }
- return level;
- };
- if (typeof item.logLevel === "undefined") {
- item.logLevel = getLogLevelForProxy(
- compilerOptions.infrastructureLogging
- ? compilerOptions.infrastructureLogging.level
- : "info"
- );
- }
- if (typeof item.logProvider === "undefined") {
- item.logProvider = () => this.logger;
- }
- return item;
- });
- }
- if (typeof options.setupExitSignals === "undefined") {
- options.setupExitSignals = true;
- }
- if (typeof options.static === "undefined") {
- options.static = [getStaticItem()];
- } else if (typeof options.static === "boolean") {
- options.static = options.static ? [getStaticItem()] : false;
- } else if (typeof options.static === "string") {
- options.static = [getStaticItem(options.static)];
- } else if (Array.isArray(options.static)) {
- options.static = options.static.map((item) => getStaticItem(item));
- } else {
- options.static = [getStaticItem(options.static)];
- }
- if (typeof options.watchFiles === "string") {
- options.watchFiles = [
- { paths: options.watchFiles, options: getWatchOptions() },
- ];
- } else if (
- typeof options.watchFiles === "object" &&
- options.watchFiles !== null &&
- !Array.isArray(options.watchFiles)
- ) {
- options.watchFiles = [
- {
- paths: options.watchFiles.paths,
- options: getWatchOptions(options.watchFiles.options || {}),
- },
- ];
- } else if (Array.isArray(options.watchFiles)) {
- options.watchFiles = options.watchFiles.map((item) => {
- if (typeof item === "string") {
- return { paths: item, options: getWatchOptions() };
- }
- return {
- paths: item.paths,
- options: getWatchOptions(item.options || {}),
- };
- });
- } else {
- options.watchFiles = [];
- }
- const defaultWebSocketServerType = "ws";
- const defaultWebSocketServerOptions = { path: "/ws" };
- if (typeof options.webSocketServer === "undefined") {
- options.webSocketServer = {
- type: defaultWebSocketServerType,
- options: defaultWebSocketServerOptions,
- };
- } else if (
- typeof options.webSocketServer === "boolean" &&
- !options.webSocketServer
- ) {
- options.webSocketServer = false;
- } else if (
- typeof options.webSocketServer === "string" ||
- typeof options.webSocketServer === "function"
- ) {
- options.webSocketServer = {
- type: options.webSocketServer,
- options: defaultWebSocketServerOptions,
- };
- } else {
- options.webSocketServer = {
- type:
- /** @type {WebSocketServerConfiguration} */
- (options.webSocketServer).type || defaultWebSocketServerType,
- options: {
- ...defaultWebSocketServerOptions,
- .../** @type {WebSocketServerConfiguration} */
- (options.webSocketServer).options,
- },
- };
- const webSocketServer =
- /** @type {{ type: WebSocketServerConfiguration["type"], options: NonNullable<WebSocketServerConfiguration["options"]> }} */
- (options.webSocketServer);
- if (typeof webSocketServer.options.port === "string") {
- webSocketServer.options.port = Number(webSocketServer.options.port);
- }
- }
- }
- /**
- * @private
- * @returns {string}
- */
- getClientTransport() {
- let clientImplementation;
- let clientImplementationFound = true;
- const isKnownWebSocketServerImplementation =
- this.options.webSocketServer &&
- typeof (
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type
- ) === "string" &&
- // @ts-ignore
- (this.options.webSocketServer.type === "ws" ||
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type === "sockjs");
- let clientTransport;
- if (this.options.client) {
- if (
- typeof (
- /** @type {ClientConfiguration} */
- (this.options.client).webSocketTransport
- ) !== "undefined"
- ) {
- clientTransport =
- /** @type {ClientConfiguration} */
- (this.options.client).webSocketTransport;
- } else if (isKnownWebSocketServerImplementation) {
- clientTransport =
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type;
- } else {
- clientTransport = "ws";
- }
- } else {
- clientTransport = "ws";
- }
- switch (typeof clientTransport) {
- case "string":
- // could be 'sockjs', 'ws', or a path that should be required
- if (clientTransport === "sockjs") {
- clientImplementation = require.resolve(
- "../client/clients/SockJSClient"
- );
- } else if (clientTransport === "ws") {
- clientImplementation = require.resolve(
- "../client/clients/WebSocketClient"
- );
- } else {
- try {
- clientImplementation = require.resolve(clientTransport);
- } catch (e) {
- clientImplementationFound = false;
- }
- }
- break;
- default:
- clientImplementationFound = false;
- }
- if (!clientImplementationFound) {
- throw new Error(
- `${
- !isKnownWebSocketServerImplementation
- ? "When you use custom web socket implementation you must explicitly specify client.webSocketTransport. "
- : ""
- }client.webSocketTransport must be a string denoting a default implementation (e.g. 'sockjs', 'ws') or a full path to a JS file via require.resolve(...) which exports a class `
- );
- }
- return /** @type {string} */ (clientImplementation);
- }
- /**
- * @private
- * @returns {string}
- */
- getServerTransport() {
- let implementation;
- let implementationFound = true;
- switch (
- typeof (
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type
- )
- ) {
- case "string":
- // Could be 'sockjs', in the future 'ws', or a path that should be required
- if (
- /** @type {WebSocketServerConfiguration} */ (
- this.options.webSocketServer
- ).type === "sockjs"
- ) {
- implementation = require("./servers/SockJSServer");
- } else if (
- /** @type {WebSocketServerConfiguration} */ (
- this.options.webSocketServer
- ).type === "ws"
- ) {
- implementation = require("./servers/WebsocketServer");
- } else {
- try {
- // eslint-disable-next-line import/no-dynamic-require
- implementation = require(/** @type {WebSocketServerConfiguration} */ (
- this.options.webSocketServer
- ).type);
- } catch (error) {
- implementationFound = false;
- }
- }
- break;
- case "function":
- implementation = /** @type {WebSocketServerConfiguration} */ (
- this.options.webSocketServer
- ).type;
- break;
- default:
- implementationFound = false;
- }
- if (!implementationFound) {
- throw new Error(
- "webSocketServer (webSocketServer.type) must be a string denoting a default implementation (e.g. 'ws', 'sockjs'), a full path to " +
- "a JS file which exports a class extending BaseServer (webpack-dev-server/lib/servers/BaseServer.js) " +
- "via require.resolve(...), or the class itself which extends BaseServer"
- );
- }
- return implementation;
- }
- /**
- * @private
- * @returns {void}
- */
- setupProgressPlugin() {
- const { ProgressPlugin } =
- /** @type {MultiCompiler}*/
- (this.compiler).compilers
- ? /** @type {MultiCompiler}*/ (this.compiler).compilers[0].webpack
- : /** @type {Compiler}*/ (this.compiler).webpack ||
- // TODO remove me after drop webpack v4
- require("webpack");
- new ProgressPlugin(
- /**
- * @param {number} percent
- * @param {string} msg
- * @param {string} addInfo
- * @param {string} pluginName
- */
- (percent, msg, addInfo, pluginName) => {
- percent = Math.floor(percent * 100);
- if (percent === 100) {
- msg = "Compilation completed";
- }
- if (addInfo) {
- msg = `${msg} (${addInfo})`;
- }
- if (this.webSocketServer) {
- this.sendMessage(this.webSocketServer.clients, "progress-update", {
- percent,
- msg,
- pluginName,
- });
- }
- if (this.server) {
- this.server.emit("progress-update", { percent, msg, pluginName });
- }
- }
- ).apply(this.compiler);
- }
- /**
- * @private
- * @returns {Promise<void>}
- */
- async initialize() {
- if (this.options.webSocketServer) {
- const compilers =
- /** @type {MultiCompiler} */
- (this.compiler).compilers || [this.compiler];
- compilers.forEach((compiler) => {
- this.addAdditionalEntries(compiler);
- const webpack = compiler.webpack || require("webpack");
- new webpack.ProvidePlugin({
- __webpack_dev_server_client__: this.getClientTransport(),
- }).apply(compiler);
- // TODO remove after drop webpack v4 support
- compiler.options.plugins = compiler.options.plugins || [];
- if (this.options.hot) {
- const HMRPluginExists = compiler.options.plugins.find(
- (p) => p.constructor === webpack.HotModuleReplacementPlugin
- );
- if (HMRPluginExists) {
- this.logger.warn(
- `"hot: true" automatically applies HMR plugin, you don't have to add it manually to your webpack configuration.`
- );
- } else {
- // Apply the HMR plugin
- const plugin = new webpack.HotModuleReplacementPlugin();
- plugin.apply(compiler);
- }
- }
- });
- if (
- this.options.client &&
- /** @type {ClientConfiguration} */ (this.options.client).progress
- ) {
- this.setupProgressPlugin();
- }
- }
- this.setupHooks();
- this.setupApp();
- this.setupHostHeaderCheck();
- this.setupDevMiddleware();
- // Should be after `webpack-dev-middleware`, otherwise other middlewares might rewrite response
- this.setupBuiltInRoutes();
- this.setupWatchFiles();
- this.setupWatchStaticFiles();
- this.setupMiddlewares();
- this.createServer();
- if (this.options.setupExitSignals) {
- const signals = ["SIGINT", "SIGTERM"];
- let needForceShutdown = false;
- signals.forEach((signal) => {
- const listener = () => {
- if (needForceShutdown) {
- process.exit();
- }
- this.logger.info(
- "Gracefully shutting down. To force exit, press ^C again. Please wait..."
- );
- needForceShutdown = true;
- this.stopCallback(() => {
- if (typeof this.compiler.close === "function") {
- this.compiler.close(() => {
- process.exit();
- });
- } else {
- process.exit();
- }
- });
- };
- this.listeners.push({ name: signal, listener });
- process.on(signal, listener);
- });
- }
- // Proxy WebSocket without the initial http request
- // https://github.com/chimurai/http-proxy-middleware#external-websocket-upgrade
- /** @type {RequestHandler[]} */
- (this.webSocketProxies).forEach((webSocketProxy) => {
- /** @type {import("http").Server} */
- (this.server).on(
- "upgrade",
- /** @type {RequestHandler & { upgrade: NonNullable<RequestHandler["upgrade"]> }} */
- (webSocketProxy).upgrade
- );
- }, this);
- }
- /**
- * @private
- * @returns {void}
- */
- setupApp() {
- /** @type {import("express").Application | undefined}*/
- // eslint-disable-next-line new-cap
- this.app = new /** @type {any} */ (express)();
- }
- /**
- * @private
- * @param {Stats | MultiStats} statsObj
- * @returns {StatsCompilation}
- */
- getStats(statsObj) {
- const stats = Server.DEFAULT_STATS;
- const compilerOptions = this.getCompilerOptions();
- // @ts-ignore
- if (compilerOptions.stats && compilerOptions.stats.warningsFilter) {
- // @ts-ignore
- stats.warningsFilter = compilerOptions.stats.warningsFilter;
- }
- return statsObj.toJson(stats);
- }
- /**
- * @private
- * @returns {void}
- */
- setupHooks() {
- this.compiler.hooks.invalid.tap("webpack-dev-server", () => {
- if (this.webSocketServer) {
- this.sendMessage(this.webSocketServer.clients, "invalid");
- }
- });
- this.compiler.hooks.done.tap(
- "webpack-dev-server",
- /**
- * @param {Stats | MultiStats} stats
- */
- (stats) => {
- if (this.webSocketServer) {
- this.sendStats(this.webSocketServer.clients, this.getStats(stats));
- }
- /**
- * @private
- * @type {Stats | MultiStats}
- */
- this.stats = stats;
- }
- );
- }
- /**
- * @private
- * @returns {void}
- */
- setupHostHeaderCheck() {
- /** @type {import("express").Application} */
- (this.app).all(
- "*",
- /**
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- * @returns {void}
- */
- (req, res, next) => {
- if (
- this.checkHeader(
- /** @type {{ [key: string]: string | undefined }} */
- (req.headers),
- "host"
- )
- ) {
- return next();
- }
- res.send("Invalid Host header");
- }
- );
- }
- /**
- * @private
- * @returns {void}
- */
- setupDevMiddleware() {
- const webpackDevMiddleware = require("webpack-dev-middleware");
- // middleware for serving webpack bundle
- this.middleware = webpackDevMiddleware(
- this.compiler,
- this.options.devMiddleware
- );
- }
- /**
- * @private
- * @returns {void}
- */
- setupBuiltInRoutes() {
- const { app, middleware } = this;
- /** @type {import("express").Application} */
- (app).get(
- "/__webpack_dev_server__/sockjs.bundle.js",
- /**
- * @param {Request} req
- * @param {Response} res
- * @returns {void}
- */
- (req, res) => {
- res.setHeader("Content-Type", "application/javascript");
- const { createReadStream } = fs;
- const clientPath = path.join(__dirname, "..", "client");
- createReadStream(
- path.join(clientPath, "modules/sockjs-client/index.js")
- ).pipe(res);
- }
- );
- /** @type {import("express").Application} */
- (app).get(
- "/webpack-dev-server/invalidate",
- /**
- * @param {Request} _req
- * @param {Response} res
- * @returns {void}
- */
- (_req, res) => {
- this.invalidate();
- res.end();
- }
- );
- /** @type {import("express").Application} */
- (app).get(
- "/webpack-dev-server",
- /**
- * @param {Request} req
- * @param {Response} res
- * @returns {void}
- */
- (req, res) => {
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (middleware).waitUntilValid((stats) => {
- res.setHeader("Content-Type", "text/html");
- res.write(
- '<!DOCTYPE html><html><head><meta charset="utf-8"/></head><body>'
- );
- const statsForPrint =
- typeof (/** @type {MultiStats} */ (stats).stats) !== "undefined"
- ? /** @type {MultiStats} */ (stats).toJson().children
- : [/** @type {Stats} */ (stats).toJson()];
- res.write(`<h1>Assets Report:</h1>`);
- /**
- * @type {StatsCompilation[]}
- */
- (statsForPrint).forEach((item, index) => {
- res.write("<div>");
- const name =
- // eslint-disable-next-line no-nested-ternary
- typeof item.name !== "undefined"
- ? item.name
- : /** @type {MultiStats} */ (stats).stats
- ? `unnamed[${index}]`
- : "unnamed";
- res.write(`<h2>Compilation: ${name}</h2>`);
- res.write("<ul>");
- const publicPath =
- item.publicPath === "auto" ? "" : item.publicPath;
- for (const asset of /** @type {NonNullable<StatsCompilation["assets"]>} */ (
- item.assets
- )) {
- const assetName = asset.name;
- const assetURL = `${publicPath}${assetName}`;
- res.write(
- `<li>
- <strong><a href="${assetURL}" target="_blank">${assetName}</a></strong>
- </li>`
- );
- }
- res.write("</ul>");
- res.write("</div>");
- });
- res.end("</body></html>");
- });
- }
- );
- }
- /**
- * @private
- * @returns {void}
- */
- setupWatchStaticFiles() {
- if (/** @type {NormalizedStatic[]} */ (this.options.static).length > 0) {
- /** @type {NormalizedStatic[]} */
- (this.options.static).forEach((staticOption) => {
- if (staticOption.watch) {
- this.watchFiles(staticOption.directory, staticOption.watch);
- }
- });
- }
- }
- /**
- * @private
- * @returns {void}
- */
- setupWatchFiles() {
- const { watchFiles } = this.options;
- if (/** @type {WatchFiles[]} */ (watchFiles).length > 0) {
- /** @type {WatchFiles[]} */
- (watchFiles).forEach((item) => {
- this.watchFiles(item.paths, item.options);
- });
- }
- }
- /**
- * @private
- * @returns {void}
- */
- setupMiddlewares() {
- /**
- * @type {Array<Middleware>}
- */
- let middlewares = [];
- // compress is placed last and uses unshift so that it will be the first middleware used
- if (this.options.compress) {
- const compression = require("compression");
- middlewares.push({ name: "compression", middleware: compression() });
- }
- if (typeof this.options.onBeforeSetupMiddleware === "function") {
- this.options.onBeforeSetupMiddleware(this);
- }
- if (typeof this.options.headers !== "undefined") {
- middlewares.push({
- name: "set-headers",
- path: "*",
- middleware: this.setHeaders.bind(this),
- });
- }
- middlewares.push({
- name: "webpack-dev-middleware",
- middleware:
- /** @type {import("webpack-dev-middleware").Middleware<Request, Response>}*/
- (this.middleware),
- });
- if (this.options.proxy) {
- const { createProxyMiddleware } = require("http-proxy-middleware");
- /**
- * @param {ProxyConfigArrayItem} proxyConfig
- * @returns {RequestHandler | undefined}
- */
- const getProxyMiddleware = (proxyConfig) => {
- // It is possible to use the `bypass` method without a `target` or `router`.
- // However, the proxy middleware has no use in this case, and will fail to instantiate.
- if (proxyConfig.target) {
- const context = proxyConfig.context || proxyConfig.path;
- return createProxyMiddleware(
- /** @type {string} */ (context),
- proxyConfig
- );
- }
- if (proxyConfig.router) {
- return createProxyMiddleware(proxyConfig);
- }
- };
- /**
- * Assume a proxy configuration specified as:
- * proxy: [
- * {
- * context: "value",
- * ...options,
- * },
- * // or:
- * function() {
- * return {
- * context: "context",
- * ...options,
- * };
- * }
- * ]
- */
- /** @type {ProxyConfigArray} */
- (this.options.proxy).forEach((proxyConfigOrCallback) => {
- /**
- * @type {RequestHandler}
- */
- let proxyMiddleware;
- let proxyConfig =
- typeof proxyConfigOrCallback === "function"
- ? proxyConfigOrCallback()
- : proxyConfigOrCallback;
- proxyMiddleware =
- /** @type {RequestHandler} */
- (getProxyMiddleware(proxyConfig));
- if (proxyConfig.ws) {
- this.webSocketProxies.push(proxyMiddleware);
- }
- /**
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- * @returns {Promise<void>}
- */
- const handler = async (req, res, next) => {
- if (typeof proxyConfigOrCallback === "function") {
- const newProxyConfig = proxyConfigOrCallback(req, res, next);
- if (newProxyConfig !== proxyConfig) {
- proxyConfig = newProxyConfig;
- proxyMiddleware =
- /** @type {RequestHandler} */
- (getProxyMiddleware(proxyConfig));
- }
- }
- // - Check if we have a bypass function defined
- // - In case the bypass function is defined we'll retrieve the
- // bypassUrl from it otherwise bypassUrl would be null
- // TODO remove in the next major in favor `context` and `router` options
- const isByPassFuncDefined = typeof proxyConfig.bypass === "function";
- const bypassUrl = isByPassFuncDefined
- ? await /** @type {ByPass} */ (proxyConfig.bypass)(
- req,
- res,
- proxyConfig
- )
- : null;
- if (typeof bypassUrl === "boolean") {
- // skip the proxy
- // @ts-ignore
- req.url = null;
- next();
- } else if (typeof bypassUrl === "string") {
- // byPass to that url
- req.url = bypassUrl;
- next();
- } else if (proxyMiddleware) {
- return proxyMiddleware(req, res, next);
- } else {
- next();
- }
- };
- middlewares.push({
- name: "http-proxy-middleware",
- middleware: handler,
- });
- // Also forward error requests to the proxy so it can handle them.
- middlewares.push({
- name: "http-proxy-middleware-error-handler",
- middleware:
- /**
- * @param {Error} error
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- * @returns {any}
- */
- (error, req, res, next) => handler(req, res, next),
- });
- });
- middlewares.push({
- name: "webpack-dev-middleware",
- middleware:
- /** @type {import("webpack-dev-middleware").Middleware<Request, Response>}*/
- (this.middleware),
- });
- }
- if (/** @type {NormalizedStatic[]} */ (this.options.static).length > 0) {
- /** @type {NormalizedStatic[]} */
- (this.options.static).forEach((staticOption) => {
- staticOption.publicPath.forEach((publicPath) => {
- middlewares.push({
- name: "express-static",
- path: publicPath,
- middleware: express.static(
- staticOption.directory,
- staticOption.staticOptions
- ),
- });
- });
- });
- }
- if (this.options.historyApiFallback) {
- const connectHistoryApiFallback = require("connect-history-api-fallback");
- const { historyApiFallback } = this.options;
- if (
- typeof (
- /** @type {ConnectHistoryApiFallbackOptions} */
- (historyApiFallback).logger
- ) === "undefined" &&
- !(
- /** @type {ConnectHistoryApiFallbackOptions} */
- (historyApiFallback).verbose
- )
- ) {
- // @ts-ignore
- historyApiFallback.logger = this.logger.log.bind(
- this.logger,
- "[connect-history-api-fallback]"
- );
- }
- // Fall back to /index.html if nothing else matches.
- middlewares.push({
- name: "connect-history-api-fallback",
- middleware: connectHistoryApiFallback(
- /** @type {ConnectHistoryApiFallbackOptions} */
- (historyApiFallback)
- ),
- });
- // include our middleware to ensure
- // it is able to handle '/index.html' request after redirect
- middlewares.push({
- name: "webpack-dev-middleware",
- middleware:
- /** @type {import("webpack-dev-middleware").Middleware<Request, Response>}*/
- (this.middleware),
- });
- if (/** @type {NormalizedStatic[]} */ (this.options.static).length > 0) {
- /** @type {NormalizedStatic[]} */
- (this.options.static).forEach((staticOption) => {
- staticOption.publicPath.forEach((publicPath) => {
- middlewares.push({
- name: "express-static",
- path: publicPath,
- middleware: express.static(
- staticOption.directory,
- staticOption.staticOptions
- ),
- });
- });
- });
- }
- }
- if (/** @type {NormalizedStatic[]} */ (this.options.static).length > 0) {
- const serveIndex = require("serve-index");
- /** @type {NormalizedStatic[]} */
- (this.options.static).forEach((staticOption) => {
- staticOption.publicPath.forEach((publicPath) => {
- if (staticOption.serveIndex) {
- middlewares.push({
- name: "serve-index",
- path: publicPath,
- /**
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- * @returns {void}
- */
- middleware: (req, res, next) => {
- // serve-index doesn't fallthrough non-get/head request to next middleware
- if (req.method !== "GET" && req.method !== "HEAD") {
- return next();
- }
- serveIndex(
- staticOption.directory,
- /** @type {ServeIndexOptions} */
- (staticOption.serveIndex)
- )(req, res, next);
- },
- });
- }
- });
- });
- }
- if (this.options.magicHtml) {
- middlewares.push({
- name: "serve-magic-html",
- middleware: this.serveMagicHtml.bind(this),
- });
- }
- if (typeof this.options.setupMiddlewares === "function") {
- middlewares = this.options.setupMiddlewares(middlewares, this);
- }
- middlewares.forEach((middleware) => {
- if (typeof middleware === "function") {
- /** @type {import("express").Application} */
- (this.app).use(middleware);
- } else if (typeof middleware.path !== "undefined") {
- /** @type {import("express").Application} */
- (this.app).use(middleware.path, middleware.middleware);
- } else {
- /** @type {import("express").Application} */
- (this.app).use(middleware.middleware);
- }
- });
- if (typeof this.options.onAfterSetupMiddleware === "function") {
- this.options.onAfterSetupMiddleware(this);
- }
- }
- /**
- * @private
- * @returns {void}
- */
- createServer() {
- const { type, options } = /** @type {ServerConfiguration} */ (
- this.options.server
- );
- /** @type {import("http").Server | undefined | null} */
- // eslint-disable-next-line import/no-dynamic-require
- this.server = require(/** @type {string} */ (type)).createServer(
- options,
- this.app
- );
- /** @type {import("http").Server} */
- (this.server).on(
- "connection",
- /**
- * @param {Socket} socket
- */
- (socket) => {
- // Add socket to list
- this.sockets.push(socket);
- socket.once("close", () => {
- // Remove socket from list
- this.sockets.splice(this.sockets.indexOf(socket), 1);
- });
- }
- );
- /** @type {import("http").Server} */
- (this.server).on(
- "error",
- /**
- * @param {Error} error
- */
- (error) => {
- throw error;
- }
- );
- }
- /**
- * @private
- * @returns {void}
- */
- // TODO: remove `--web-socket-server` in favor of `--web-socket-server-type`
- createWebSocketServer() {
- /** @type {WebSocketServerImplementation | undefined | null} */
- this.webSocketServer = new /** @type {any} */ (this.getServerTransport())(
- this
- );
- /** @type {WebSocketServerImplementation} */
- (this.webSocketServer).implementation.on(
- "connection",
- /**
- * @param {ClientConnection} client
- * @param {IncomingMessage} request
- */
- (client, request) => {
- /** @type {{ [key: string]: string | undefined } | undefined} */
- const headers =
- // eslint-disable-next-line no-nested-ternary
- typeof request !== "undefined"
- ? /** @type {{ [key: string]: string | undefined }} */
- (request.headers)
- : typeof (
- /** @type {import("sockjs").Connection} */ (client).headers
- ) !== "undefined"
- ? /** @type {import("sockjs").Connection} */ (client).headers
- : // eslint-disable-next-line no-undefined
- undefined;
- if (!headers) {
- this.logger.warn(
- 'webSocketServer implementation must pass headers for the "connection" event'
- );
- }
- if (
- !headers ||
- !this.checkHeader(headers, "host") ||
- !this.checkHeader(headers, "origin")
- ) {
- this.sendMessage([client], "error", "Invalid Host/Origin header");
- // With https enabled, the sendMessage above is encrypted asynchronously so not yet sent
- // Terminate would prevent it sending, so use close to allow it to be sent
- client.close();
- return;
- }
- if (this.options.hot === true || this.options.hot === "only") {
- this.sendMessage([client], "hot");
- }
- if (this.options.liveReload) {
- this.sendMessage([client], "liveReload");
- }
- if (
- this.options.client &&
- /** @type {ClientConfiguration} */
- (this.options.client).progress
- ) {
- this.sendMessage(
- [client],
- "progress",
- /** @type {ClientConfiguration} */
- (this.options.client).progress
- );
- }
- if (
- this.options.client &&
- /** @type {ClientConfiguration} */ (this.options.client).reconnect
- ) {
- this.sendMessage(
- [client],
- "reconnect",
- /** @type {ClientConfiguration} */
- (this.options.client).reconnect
- );
- }
- if (
- this.options.client &&
- /** @type {ClientConfiguration} */
- (this.options.client).overlay
- ) {
- this.sendMessage(
- [client],
- "overlay",
- /** @type {ClientConfiguration} */
- (this.options.client).overlay
- );
- }
- if (!this.stats) {
- return;
- }
- this.sendStats([client], this.getStats(this.stats), true);
- }
- );
- }
- /**
- * @private
- * @param {string} defaultOpenTarget
- * @returns {void}
- */
- openBrowser(defaultOpenTarget) {
- const open = require("open");
- Promise.all(
- /** @type {NormalizedOpen[]} */
- (this.options.open).map((item) => {
- /**
- * @type {string}
- */
- let openTarget;
- if (item.target === "<url>") {
- openTarget = defaultOpenTarget;
- } else {
- openTarget = Server.isAbsoluteURL(item.target)
- ? item.target
- : new URL(item.target, defaultOpenTarget).toString();
- }
- return open(openTarget, item.options).catch(() => {
- this.logger.warn(
- `Unable to open "${openTarget}" page${
- item.options.app
- ? ` in "${
- /** @type {import("open").App} */
- (item.options.app).name
- }" app${
- /** @type {import("open").App} */
- (item.options.app).arguments
- ? ` with "${
- /** @type {import("open").App} */
- (item.options.app).arguments.join(" ")
- }" arguments`
- : ""
- }`
- : ""
- }. If you are running in a headless environment, please do not use the "open" option or related flags like "--open", "--open-target", and "--open-app".`
- );
- });
- })
- );
- }
- /**
- * @private
- * @returns {void}
- */
- runBonjour() {
- const { Bonjour } = require("bonjour-service");
- /**
- * @private
- * @type {Bonjour | undefined}
- */
- this.bonjour = new Bonjour();
- this.bonjour.publish({
- // @ts-expect-error
- name: `Webpack Dev Server ${os.hostname()}:${this.options.port}`,
- // @ts-expect-error
- port: /** @type {number} */ (this.options.port),
- // @ts-expect-error
- type:
- /** @type {ServerConfiguration} */
- (this.options.server).type === "http" ? "http" : "https",
- subtypes: ["webpack"],
- .../** @type {BonjourOptions} */ (this.options.bonjour),
- });
- }
- /**
- * @private
- * @returns {void}
- */
- stopBonjour(callback = () => {}) {
- /** @type {Bonjour} */
- (this.bonjour).unpublishAll(() => {
- /** @type {Bonjour} */
- (this.bonjour).destroy();
- if (callback) {
- callback();
- }
- });
- }
- /**
- * @private
- * @returns {void}
- */
- logStatus() {
- const { isColorSupported, cyan, red } = require("colorette");
- /**
- * @param {Compiler["options"]} compilerOptions
- * @returns {boolean}
- */
- const getColorsOption = (compilerOptions) => {
- /**
- * @type {boolean}
- */
- let colorsEnabled;
- if (
- compilerOptions.stats &&
- typeof (/** @type {StatsOptions} */ (compilerOptions.stats).colors) !==
- "undefined"
- ) {
- colorsEnabled =
- /** @type {boolean} */
- (/** @type {StatsOptions} */ (compilerOptions.stats).colors);
- } else {
- colorsEnabled = isColorSupported;
- }
- return colorsEnabled;
- };
- const colors = {
- /**
- * @param {boolean} useColor
- * @param {string} msg
- * @returns {string}
- */
- info(useColor, msg) {
- if (useColor) {
- return cyan(msg);
- }
- return msg;
- },
- /**
- * @param {boolean} useColor
- * @param {string} msg
- * @returns {string}
- */
- error(useColor, msg) {
- if (useColor) {
- return red(msg);
- }
- return msg;
- },
- };
- const useColor = getColorsOption(this.getCompilerOptions());
- if (this.options.ipc) {
- this.logger.info(
- `Project is running at: "${
- /** @type {import("http").Server} */
- (this.server).address()
- }"`
- );
- } else {
- const protocol =
- /** @type {ServerConfiguration} */
- (this.options.server).type === "http" ? "http" : "https";
- const { address, port } =
- /** @type {import("net").AddressInfo} */
- (
- /** @type {import("http").Server} */
- (this.server).address()
- );
- /**
- * @param {string} newHostname
- * @returns {string}
- */
- const prettyPrintURL = (newHostname) =>
- url.format({ protocol, hostname: newHostname, port, pathname: "/" });
- let server;
- let localhost;
- let loopbackIPv4;
- let loopbackIPv6;
- let networkUrlIPv4;
- let networkUrlIPv6;
- if (this.options.host) {
- if (this.options.host === "localhost") {
- localhost = prettyPrintURL("localhost");
- } else {
- let isIP;
- try {
- isIP = ipaddr.parse(this.options.host);
- } catch (error) {
- // Ignore
- }
- if (!isIP) {
- server = prettyPrintURL(this.options.host);
- }
- }
- }
- const parsedIP = ipaddr.parse(address);
- if (parsedIP.range() === "unspecified") {
- localhost = prettyPrintURL("localhost");
- const networkIPv4 = Server.internalIPSync("v4");
- if (networkIPv4) {
- networkUrlIPv4 = prettyPrintURL(networkIPv4);
- }
- const networkIPv6 = Server.internalIPSync("v6");
- if (networkIPv6) {
- networkUrlIPv6 = prettyPrintURL(networkIPv6);
- }
- } else if (parsedIP.range() === "loopback") {
- if (parsedIP.kind() === "ipv4") {
- loopbackIPv4 = prettyPrintURL(parsedIP.toString());
- } else if (parsedIP.kind() === "ipv6") {
- loopbackIPv6 = prettyPrintURL(parsedIP.toString());
- }
- } else {
- networkUrlIPv4 =
- parsedIP.kind() === "ipv6" &&
- /** @type {IPv6} */
- (parsedIP).isIPv4MappedAddress()
- ? prettyPrintURL(
- /** @type {IPv6} */
- (parsedIP).toIPv4Address().toString()
- )
- : prettyPrintURL(address);
- if (parsedIP.kind() === "ipv6") {
- networkUrlIPv6 = prettyPrintURL(address);
- }
- }
- this.logger.info("Project is running at:");
- if (server) {
- this.logger.info(`Server: ${colors.info(useColor, server)}`);
- }
- if (localhost || loopbackIPv4 || loopbackIPv6) {
- const loopbacks = [];
- if (localhost) {
- loopbacks.push([colors.info(useColor, localhost)]);
- }
- if (loopbackIPv4) {
- loopbacks.push([colors.info(useColor, loopbackIPv4)]);
- }
- if (loopbackIPv6) {
- loopbacks.push([colors.info(useColor, loopbackIPv6)]);
- }
- this.logger.info(`Loopback: ${loopbacks.join(", ")}`);
- }
- if (networkUrlIPv4) {
- this.logger.info(
- `On Your Network (IPv4): ${colors.info(useColor, networkUrlIPv4)}`
- );
- }
- if (networkUrlIPv6) {
- this.logger.info(
- `On Your Network (IPv6): ${colors.info(useColor, networkUrlIPv6)}`
- );
- }
- if (/** @type {NormalizedOpen[]} */ (this.options.open).length > 0) {
- const openTarget = prettyPrintURL(this.options.host || "localhost");
- this.openBrowser(openTarget);
- }
- }
- if (/** @type {NormalizedStatic[]} */ (this.options.static).length > 0) {
- this.logger.info(
- `Content not from webpack is served from '${colors.info(
- useColor,
- /** @type {NormalizedStatic[]} */
- (this.options.static)
- .map((staticOption) => staticOption.directory)
- .join(", ")
- )}' directory`
- );
- }
- if (this.options.historyApiFallback) {
- this.logger.info(
- `404s will fallback to '${colors.info(
- useColor,
- /** @type {ConnectHistoryApiFallbackOptions} */ (
- this.options.historyApiFallback
- ).index || "/index.html"
- )}'`
- );
- }
- if (this.options.bonjour) {
- const bonjourProtocol =
- /** @type {BonjourOptions} */
- (this.options.bonjour).type ||
- /** @type {ServerConfiguration} */
- (this.options.server).type === "http"
- ? "http"
- : "https";
- this.logger.info(
- `Broadcasting "${bonjourProtocol}" with subtype of "webpack" via ZeroConf DNS (Bonjour)`
- );
- }
- }
- /**
- * @private
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- */
- setHeaders(req, res, next) {
- let { headers } = this.options;
- if (headers) {
- if (typeof headers === "function") {
- headers = headers(
- req,
- res,
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (this.middleware).context
- );
- }
- /**
- * @type {{key: string, value: string}[]}
- */
- const allHeaders = [];
- if (!Array.isArray(headers)) {
- // eslint-disable-next-line guard-for-in
- for (const name in headers) {
- // @ts-ignore
- allHeaders.push({ key: name, value: headers[name] });
- }
- headers = allHeaders;
- }
- headers.forEach(
- /**
- * @param {{key: string, value: any}} header
- */
- (header) => {
- res.setHeader(header.key, header.value);
- }
- );
- }
- next();
- }
- /**
- * @private
- * @param {{ [key: string]: string | undefined }} headers
- * @param {string} headerToCheck
- * @returns {boolean}
- */
- checkHeader(headers, headerToCheck) {
- // allow user to opt out of this security check, at their own risk
- // by explicitly enabling allowedHosts
- if (this.options.allowedHosts === "all") {
- return true;
- }
- // get the Host header and extract hostname
- // we don't care about port not matching
- const hostHeader = headers[headerToCheck];
- if (!hostHeader) {
- return false;
- }
- if (/^(file|.+-extension):/i.test(hostHeader)) {
- return true;
- }
- // use the node url-parser to retrieve the hostname from the host-header.
- const hostname = url.parse(
- // if hostHeader doesn't have scheme, add // for parsing.
- /^(.+:)?\/\//.test(hostHeader) ? hostHeader : `//${hostHeader}`,
- false,
- true
- ).hostname;
- // always allow requests with explicit IPv4 or IPv6-address.
- // A note on IPv6 addresses:
- // hostHeader will always contain the brackets denoting
- // an IPv6-address in URLs,
- // these are removed from the hostname in url.parse(),
- // so we have the pure IPv6-address in hostname.
- // always allow localhost host, for convenience (hostname === 'localhost')
- // allow hostname of listening address (hostname === this.options.host)
- const isValidHostname =
- (hostname !== null && ipaddr.IPv4.isValid(hostname)) ||
- (hostname !== null && ipaddr.IPv6.isValid(hostname)) ||
- hostname === "localhost" ||
- hostname === this.options.host;
- if (isValidHostname) {
- return true;
- }
- const { allowedHosts } = this.options;
- // always allow localhost host, for convenience
- // allow if hostname is in allowedHosts
- if (Array.isArray(allowedHosts) && allowedHosts.length > 0) {
- for (let hostIdx = 0; hostIdx < allowedHosts.length; hostIdx++) {
- const allowedHost = allowedHosts[hostIdx];
- if (allowedHost === hostname) {
- return true;
- }
- // support "." as a subdomain wildcard
- // e.g. ".example.com" will allow "example.com", "www.example.com", "subdomain.example.com", etc
- if (allowedHost[0] === ".") {
- // "example.com" (hostname === allowedHost.substring(1))
- // "*.example.com" (hostname.endsWith(allowedHost))
- if (
- hostname === allowedHost.substring(1) ||
- /** @type {string} */ (hostname).endsWith(allowedHost)
- ) {
- return true;
- }
- }
- }
- }
- // Also allow if `client.webSocketURL.hostname` provided
- if (
- this.options.client &&
- typeof (
- /** @type {ClientConfiguration} */ (this.options.client).webSocketURL
- ) !== "undefined"
- ) {
- return (
- /** @type {WebSocketURL} */
- (/** @type {ClientConfiguration} */ (this.options.client).webSocketURL)
- .hostname === hostname
- );
- }
- // disallow
- return false;
- }
- /**
- * @param {ClientConnection[]} clients
- * @param {string} type
- * @param {any} [data]
- * @param {any} [params]
- */
- // eslint-disable-next-line class-methods-use-this
- sendMessage(clients, type, data, params) {
- for (const client of clients) {
- // `sockjs` uses `1` to indicate client is ready to accept data
- // `ws` uses `WebSocket.OPEN`, but it is mean `1` too
- if (client.readyState === 1) {
- client.send(JSON.stringify({ type, data, params }));
- }
- }
- }
- /**
- * @private
- * @param {Request} req
- * @param {Response} res
- * @param {NextFunction} next
- * @returns {void}
- */
- serveMagicHtml(req, res, next) {
- if (req.method !== "GET" && req.method !== "HEAD") {
- return next();
- }
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (this.middleware).waitUntilValid(() => {
- const _path = req.path;
- try {
- const filename =
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (this.middleware).getFilenameFromUrl(`${_path}.js`);
- const isFile =
- /** @type {Compiler["outputFileSystem"] & { statSync: import("fs").StatSyncFn }}*/
- (
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (this.middleware).context.outputFileSystem
- )
- .statSync(/** @type {import("fs").PathLike} */ (filename))
- .isFile();
- if (!isFile) {
- return next();
- }
- // Serve a page that executes the javascript
- // @ts-ignore
- const queries = req._parsedUrl.search || "";
- const responsePage = `<!DOCTYPE html><html><head><meta charset="utf-8"/></head><body><script type="text/javascript" charset="utf-8" src="${_path}.js${queries}"></script></body></html>`;
- res.send(responsePage);
- } catch (error) {
- return next();
- }
- });
- }
- // Send stats to a socket or multiple sockets
- /**
- * @private
- * @param {ClientConnection[]} clients
- * @param {StatsCompilation} stats
- * @param {boolean} [force]
- */
- sendStats(clients, stats, force) {
- const shouldEmit =
- !force &&
- stats &&
- (!stats.errors || stats.errors.length === 0) &&
- (!stats.warnings || stats.warnings.length === 0) &&
- this.currentHash === stats.hash;
- if (shouldEmit) {
- this.sendMessage(clients, "still-ok");
- return;
- }
- this.currentHash = stats.hash;
- this.sendMessage(clients, "hash", stats.hash);
- if (
- /** @type {NonNullable<StatsCompilation["errors"]>} */
- (stats.errors).length > 0 ||
- /** @type {NonNullable<StatsCompilation["warnings"]>} */
- (stats.warnings).length > 0
- ) {
- const hasErrors =
- /** @type {NonNullable<StatsCompilation["errors"]>} */
- (stats.errors).length > 0;
- if (
- /** @type {NonNullable<StatsCompilation["warnings"]>} */
- (stats.warnings).length > 0
- ) {
- let params;
- if (hasErrors) {
- params = { preventReloading: true };
- }
- this.sendMessage(clients, "warnings", stats.warnings, params);
- }
- if (
- /** @type {NonNullable<StatsCompilation["errors"]>} */ (stats.errors)
- .length > 0
- ) {
- this.sendMessage(clients, "errors", stats.errors);
- }
- } else {
- this.sendMessage(clients, "ok");
- }
- }
- /**
- * @param {string | string[]} watchPath
- * @param {WatchOptions} [watchOptions]
- */
- watchFiles(watchPath, watchOptions) {
- const chokidar = require("chokidar");
- const watcher = chokidar.watch(watchPath, watchOptions);
- // disabling refreshing on changing the content
- if (this.options.liveReload) {
- watcher.on("change", (item) => {
- if (this.webSocketServer) {
- this.sendMessage(
- this.webSocketServer.clients,
- "static-changed",
- item
- );
- }
- });
- }
- this.staticWatchers.push(watcher);
- }
- /**
- * @param {import("webpack-dev-middleware").Callback} [callback]
- */
- invalidate(callback = () => {}) {
- if (this.middleware) {
- this.middleware.invalidate(callback);
- }
- }
- /**
- * @returns {Promise<void>}
- */
- async start() {
- await this.normalizeOptions();
- if (this.options.ipc) {
- await /** @type {Promise<void>} */ (
- new Promise((resolve, reject) => {
- const net = require("net");
- const socket = new net.Socket();
- socket.on(
- "error",
- /**
- * @param {Error & { code?: string }} error
- */
- (error) => {
- if (error.code === "ECONNREFUSED") {
- // No other server listening on this socket so it can be safely removed
- fs.unlinkSync(/** @type {string} */ (this.options.ipc));
- resolve();
- return;
- } else if (error.code === "ENOENT") {
- resolve();
- return;
- }
- reject(error);
- }
- );
- socket.connect(
- { path: /** @type {string} */ (this.options.ipc) },
- () => {
- throw new Error(`IPC "${this.options.ipc}" is already used`);
- }
- );
- })
- );
- } else {
- this.options.host = await Server.getHostname(
- /** @type {Host} */ (this.options.host)
- );
- this.options.port = await Server.getFreePort(
- /** @type {Port} */ (this.options.port)
- );
- }
- await this.initialize();
- const listenOptions = this.options.ipc
- ? { path: this.options.ipc }
- : { host: this.options.host, port: this.options.port };
- await /** @type {Promise<void>} */ (
- new Promise((resolve) => {
- /** @type {import("http").Server} */
- (this.server).listen(listenOptions, () => {
- resolve();
- });
- })
- );
- if (this.options.ipc) {
- // chmod 666 (rw rw rw)
- const READ_WRITE = 438;
- await fs.promises.chmod(
- /** @type {string} */ (this.options.ipc),
- READ_WRITE
- );
- }
- if (this.options.webSocketServer) {
- this.createWebSocketServer();
- }
- if (this.options.bonjour) {
- this.runBonjour();
- }
- this.logStatus();
- if (typeof this.options.onListening === "function") {
- this.options.onListening(this);
- }
- }
- /**
- * @param {(err?: Error) => void} [callback]
- */
- startCallback(callback = () => {}) {
- this.start()
- .then(() => callback(), callback)
- .catch(callback);
- }
- /**
- * @returns {Promise<void>}
- */
- async stop() {
- if (this.bonjour) {
- await /** @type {Promise<void>} */ (
- new Promise((resolve) => {
- this.stopBonjour(() => {
- resolve();
- });
- })
- );
- }
- this.webSocketProxies = [];
- await Promise.all(this.staticWatchers.map((watcher) => watcher.close()));
- this.staticWatchers = [];
- if (this.webSocketServer) {
- await /** @type {Promise<void>} */ (
- new Promise((resolve) => {
- /** @type {WebSocketServerImplementation} */
- (this.webSocketServer).implementation.close(() => {
- this.webSocketServer = null;
- resolve();
- });
- for (const client of /** @type {WebSocketServerImplementation} */ (
- this.webSocketServer
- ).clients) {
- client.terminate();
- }
- /** @type {WebSocketServerImplementation} */
- (this.webSocketServer).clients = [];
- })
- );
- }
- if (this.server) {
- await /** @type {Promise<void>} */ (
- new Promise((resolve) => {
- /** @type {import("http").Server} */
- (this.server).close(() => {
- this.server = null;
- resolve();
- });
- for (const socket of this.sockets) {
- socket.destroy();
- }
- this.sockets = [];
- })
- );
- if (this.middleware) {
- await /** @type {Promise<void>} */ (
- new Promise((resolve, reject) => {
- /** @type {import("webpack-dev-middleware").API<Request, Response>}*/
- (this.middleware).close((error) => {
- if (error) {
- reject(error);
- return;
- }
- resolve();
- });
- })
- );
- this.middleware = null;
- }
- }
- // We add listeners to signals when creating a new Server instance
- // So ensure they are removed to prevent EventEmitter memory leak warnings
- for (const item of this.listeners) {
- process.removeListener(item.name, item.listener);
- }
- }
- /**
- * @param {(err?: Error) => void} [callback]
- */
- stopCallback(callback = () => {}) {
- this.stop()
- .then(() => callback(), callback)
- .catch(callback);
- }
- // TODO remove in the next major release
- /**
- * @param {Port} port
- * @param {Host} hostname
- * @param {(err?: Error) => void} fn
- * @returns {void}
- */
- listen(port, hostname, fn) {
- util.deprecate(
- () => {},
- "'listen' is deprecated. Please use the async 'start' or 'startCallback' method.",
- "DEP_WEBPACK_DEV_SERVER_LISTEN"
- )();
- if (typeof port === "function") {
- fn = port;
- }
- if (
- typeof port !== "undefined" &&
- typeof this.options.port !== "undefined" &&
- port !== this.options.port
- ) {
- this.options.port = port;
- this.logger.warn(
- 'The "port" specified in options is different from the port passed as an argument. Will be used from arguments.'
- );
- }
- if (!this.options.port) {
- this.options.port = port;
- }
- if (
- typeof hostname !== "undefined" &&
- typeof this.options.host !== "undefined" &&
- hostname !== this.options.host
- ) {
- this.options.host = hostname;
- this.logger.warn(
- 'The "host" specified in options is different from the host passed as an argument. Will be used from arguments.'
- );
- }
- if (!this.options.host) {
- this.options.host = hostname;
- }
- this.start()
- .then(() => {
- if (fn) {
- fn.call(this.server);
- }
- })
- .catch((error) => {
- // Nothing
- if (fn) {
- fn.call(this.server, error);
- }
- });
- }
- /**
- * @param {(err?: Error) => void} [callback]
- * @returns {void}
- */
- // TODO remove in the next major release
- close(callback) {
- util.deprecate(
- () => {},
- "'close' is deprecated. Please use the async 'stop' or 'stopCallback' method.",
- "DEP_WEBPACK_DEV_SERVER_CLOSE"
- )();
- this.stop()
- .then(() => {
- if (callback) {
- callback();
- }
- })
- .catch((error) => {
- if (callback) {
- callback(error);
- }
- });
- }
- }
- module.exports = Server;
|