| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616 |
- "use strict";
- const os = require("node:os");
- const path = require("node:path");
- const url = require("node:url");
- const util = require("node:util");
- const fs = require("graceful-fs");
- const ipaddr = require("ipaddr.js");
- 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("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").Server} HTTPServer */
- /** @typedef {import("http").IncomingMessage} IncomingMessage */
- /** @typedef {import("http").ServerResponse} ServerResponse */
- /** @typedef {import("open").Options} OpenOptions */
- /** @typedef {import("express").Application} ExpressApplication */
- /** @typedef {import("express").RequestHandler} ExpressRequestHandler */
- /** @typedef {import("express").ErrorRequestHandler} ExpressErrorRequestHandler */
- /** @typedef {import("express").Request} ExpressRequest */
- /** @typedef {import("express").Response} ExpressResponse */
- // eslint-disable-next-line jsdoc/no-restricted-syntax
- /** @typedef {any} EXPECTED_ANY */
- /** @typedef {(err?: EXPECTED_ANY) => void} NextFunction */
- /** @typedef {(req: IncomingMessage, res: ServerResponse) => void} SimpleHandleFunction */
- /** @typedef {(req: IncomingMessage, res: ServerResponse, next: NextFunction) => void} NextHandleFunction */
- /** @typedef {(err: EXPECTED_ANY, req: IncomingMessage, res: ServerResponse, next: NextFunction) => void} ErrorHandleFunction */
- /** @typedef {SimpleHandleFunction | NextHandleFunction | ErrorHandleFunction} HandleFunction */
- /** @typedef {import("https").ServerOptions & { spdy?: { plain?: boolean | undefined, ssl?: boolean | undefined, 'x-forwarded-for'?: string | undefined, protocol?: string | undefined, protocols?: string[] | undefined }}} ServerOptions */
- /**
- * @template {BasicApplication} [T=ExpressApplication]
- * @typedef {T extends ExpressApplication ? ExpressRequest : IncomingMessage} Request
- */
- /**
- * @template {BasicApplication} [T=ExpressApplication]
- * @typedef {T extends ExpressApplication ? ExpressResponse : ServerResponse} Response
- */
- /**
- * @template {Request} T
- * @template {Response} U
- * @typedef {import("webpack-dev-middleware").Options<T, U>} DevMiddlewareOptions
- */
- /**
- * @template {Request} T
- * @template {Response} U
- * @typedef {import("webpack-dev-middleware").Context<T, U>} DevMiddlewareContext
- */
- /**
- * @typedef {"local-ip" | "local-ipv4" | "local-ipv6" | string} Host
- */
- /**
- * @typedef {number | string | "auto"} Port
- */
- /**
- * @typedef {object} WatchFiles
- * @property {string | string[]} paths paths
- * @property {(WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean })=} options options
- */
- /**
- * @typedef {object} Static
- * @property {string=} directory directory
- * @property {(string | string[])=} publicPath public path
- * @property {(boolean | ServeIndexOptions)=} serveIndex serve index
- * @property {ServeStaticOptions=} staticOptions static options
- * @property {(boolean | WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean })=} watch watch and watch options
- */
- /**
- * @typedef {object} NormalizedStatic
- * @property {string} directory
- * @property {string[]} publicPath
- * @property {false | ServeIndexOptions} serveIndex
- * @property {ServeStaticOptions} staticOptions
- * @property {false | WatchOptions} watch
- */
- /**
- * @template {BasicApplication} [A=ExpressApplication]
- * @template {BasicServer} [S=import("http").Server]
- * @typedef {"http" | "https" | "spdy" | "http2" | string | ((serverOptions: ServerOptions, application: A) => S)} ServerType
- */
- /**
- * @template {BasicApplication} [A=ExpressApplication]
- * @template {BasicServer} [S=import("http").Server]
- * @typedef {object} ServerConfiguration
- * @property {ServerType<A, S>=} type type
- * @property {ServerOptions=} options options
- */
- /**
- * @typedef {object} WebSocketServerConfiguration
- * @property {("sockjs" | "ws" | string | (() => WebSocketServerConfiguration))=} type type
- * @property {Record<string, EXPECTED_ANY>=} options 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 {object} OpenApp
- * @property {string=} name
- * @property {string[]=} arguments
- */
- /**
- * @typedef {object} Open
- * @property {(string | string[] | OpenApp)=} app
- * @property {(string | string[])=} target target
- */
- /**
- * @typedef {object} NormalizedOpen
- * @property {string} target
- * @property {import("open").Options} options
- */
- /**
- * @typedef {object} WebSocketURL
- * @property {string=} hostname hostname
- * @property {string=} password password
- * @property {string=} pathname pathname
- * @property {(number | string)=} port port
- * @property {string=} protocol protocol
- * @property {string=} username username
- */
- /**
- * @typedef {boolean | ((error: Error) => void)} OverlayMessageOptions
- */
- /**
- * @typedef {object} ClientConfiguration
- * @property {"log" | "info" | "warn" | "error" | "none" | "verbose"=} logging logging
- * @property {(boolean | { warnings?: OverlayMessageOptions, errors?: OverlayMessageOptions, runtimeErrors?: OverlayMessageOptions })=} overlay overlay
- * @property {boolean=} progress progress
- * @property {(boolean | number)=} reconnect reconnect
- * @property {("ws" | "sockjs" | string)=} webSocketTransport web socket transport
- * @property {(string | WebSocketURL)=} webSocketURL web socket URL
- */
- /**
- * @typedef {Array<{ key: string; value: string }> | Record<string, string | string[]>} Headers
- */
- /**
- * @template {BasicApplication} [T=ExpressApplication]
- * @typedef {T extends ExpressApplication ? ExpressRequestHandler | ExpressErrorRequestHandler : HandleFunction} MiddlewareHandler
- */
- /**
- * @typedef {{ name?: string, path?: string, middleware: MiddlewareHandler }} MiddlewareObject
- */
- /**
- * @typedef {MiddlewareObject | MiddlewareHandler } Middleware
- */
- /** @typedef {import("net").Server | import("tls").Server} BasicServer */
- /**
- * @template {BasicApplication} [A=ExpressApplication]
- * @template {BasicServer} [S=import("http").Server]
- * @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 {("auto" | "all" | string | string[])=} allowedHosts
- * @property {(boolean | ConnectHistoryApiFallbackOptions)=} historyApiFallback
- * @property {(boolean | Record<string, never> | BonjourOptions)=} bonjour
- * @property {(string | string[] | WatchFiles | Array<string | WatchFiles>)=} watchFiles
- * @property {(boolean | string | Static | Array<string | Static>)=} static
- * @property {(ServerType<A, S> | ServerConfiguration<A, S>)=} server
- * @property {(() => Promise<A>)=} app
- * @property {(boolean | "sockjs" | "ws" | string | WebSocketServerConfiguration)=} webSocketServer
- * @property {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> | undefined) => Headers))=} headers
- * @property {((devServer: Server<A, S>) => void)=} onListening
- * @property {((middlewares: Middleware[], devServer: Server<A, S>) => Middleware[])=} setupMiddlewares
- */
- if (!process.env.WEBPACK_SERVE) {
- process.env.WEBPACK_SERVE = "true";
- }
- /**
- * @template T
- * @typedef {() => T} FunctionReturning
- */
- /**
- * @template T
- * @param {FunctionReturning<T>} fn memorized function
- * @returns {FunctionReturning<T>} new function
- */
- const memoize = (fn) => {
- let cache = false;
- /** @type {T | undefined} */
- let result;
- return () => {
- if (cache) {
- return /** @type {T} */ (result);
- }
- result = fn();
- cache = true;
- // Allow to clean up memory for fn
- // and all dependent resources
- /** @type {FunctionReturning<T> | undefined} */
- (fn) = undefined;
- return /** @type {T} */ (result);
- };
- };
- const getExpress = memoize(() => require("express"));
- /**
- * @param {OverlayMessageOptions=} setting overlay settings
- * @returns {undefined | string | boolean} encoded overlay settings
- */
- const encodeOverlaySettings = (setting) =>
- typeof setting === "function"
- ? encodeURIComponent(setting.toString())
- : setting;
- // Working for overload, because typescript doesn't support this yes
- /**
- * @overload
- * @param {NextHandleFunction} fn function
- * @returns {BasicApplication} application
- */
- /**
- * @overload
- * @param {HandleFunction} fn function
- * @returns {BasicApplication} application
- */
- /**
- * @overload
- * @param {string} route route
- * @param {NextHandleFunction} fn function
- * @returns {BasicApplication} application
- */
- /**
- * @param {string} route route
- * @param {HandleFunction} fn function
- * @returns {BasicApplication} application
- */
- // eslint-disable-next-line no-unused-vars
- function useFn(route, fn) {
- return /** @type {BasicApplication} */ ({});
- }
- const DEFAULT_ALLOWED_PROTOCOLS = /^(file|.+-extension):/i;
- /**
- * @typedef {object} BasicApplication
- * @property {typeof useFn} use
- */
- /**
- * @template {BasicApplication} [A=ExpressApplication]
- * @template {BasicServer} [S=HTTPServer]
- */
- class Server {
- /**
- * @param {Configuration<A, S>} options options
- * @param {Compiler | MultiCompiler} compiler compiler
- */
- constructor(options, compiler) {
- validate(/** @type {Schema} */ (schema), options, {
- name: "Dev Server",
- baseDataPath: "options",
- });
- this.compiler = compiler;
- /**
- * @type {ReturnType<Compiler["getInfrastructureLogger"]>}
- */
- this.logger = this.compiler.getInfrastructureLogger("webpack-dev-server");
- this.options = options;
- /**
- * @type {FSWatcher[]}
- */
- this.staticWatchers = [];
- /**
- * @private
- * @type {{ name: string | symbol, listener: (...args: EXPECTED_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}
- */
- this.currentHash = undefined;
- }
- static get schema() {
- return schema;
- }
- /**
- * @private
- * @returns {StatsOptions} default stats options
- */
- static get DEFAULT_STATS() {
- return {
- all: false,
- hash: true,
- warnings: true,
- errors: true,
- errorDetails: false,
- };
- }
- /**
- * @param {string} URL url
- * @returns {boolean} true when URL is absolute, otherwise false
- */
- 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} gatewayOrFamily gateway or family
- * @param {boolean=} isInternal ip should be internal
- * @returns {string | undefined} resolved IP
- */
- static findIp(gatewayOrFamily, isInternal) {
- if (gatewayOrFamily === "v4" || gatewayOrFamily === "v6") {
- let host;
- const networks = Object.values(os.networkInterfaces())
- .flatMap((networks) => networks ?? [])
- .filter((network) => {
- if (!network || !network.address) {
- return false;
- }
- if (network.family !== `IP${gatewayOrFamily}`) {
- return false;
- }
- if (
- typeof isInternal !== "undefined" &&
- network.internal !== isInternal
- ) {
- return false;
- }
- if (gatewayOrFamily === "v6") {
- const range = ipaddr.parse(network.address).range();
- if (
- range !== "ipv4Mapped" &&
- range !== "uniqueLocal" &&
- range !== "loopback"
- ) {
- return false;
- }
- }
- return network.address;
- });
- if (networks.length > 0) {
- // Take the first network found
- host = networks[0].address;
- if (host.includes(":")) {
- host = `[${host}]`;
- }
- }
- return host;
- }
- const gatewayIp = ipaddr.parse(gatewayOrFamily);
- // 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() &&
- // eslint-disable-next-line unicorn/prefer-regexp-test
- gatewayIp.match(net)
- ) {
- return net[0].toString();
- }
- }
- }
- }
- // TODO remove me in the next major release, we have `findIp`
- /**
- * @param {"v4" | "v6"} family family
- * @returns {Promise<string | undefined>} internal API
- */
- static async internalIP(family) {
- return Server.findIp(family, false);
- }
- // TODO remove me in the next major release, we have `findIp`
- /**
- * @param {"v4" | "v6"} family family
- * @returns {string | undefined} internal IP
- */
- static internalIPSync(family) {
- return Server.findIp(family, false);
- }
- /**
- * @param {Host} hostname hostname
- * @returns {Promise<string>} resolved hostname
- */
- static async getHostname(hostname) {
- if (hostname === "local-ip") {
- return (
- Server.findIp("v4", false) || Server.findIp("v6", false) || "0.0.0.0"
- );
- } else if (hostname === "local-ipv4") {
- return Server.findIp("v4", false) || "0.0.0.0";
- } else if (hostname === "local-ipv6") {
- return Server.findIp("v6", false) || "::";
- }
- return hostname;
- }
- /**
- * @param {Port} port port
- * @param {string} host host
- * @returns {Promise<number | string>} free port
- */
- static async getFreePort(port, host) {
- if (typeof port !== "undefined" && port !== null && port !== "auto") {
- return port;
- }
- const pRetry = (await import("p-retry")).default;
- const getPort = require("./getPort");
- const basePort =
- typeof process.env.WEBPACK_DEV_SERVER_BASE_PORT !== "undefined"
- ? Number.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"
- ? Number.parseInt(process.env.WEBPACK_DEV_SERVER_PORT_RETRY, 10)
- : 3;
- return pRetry(() => getPort(basePort, host), {
- retries: defaultPortRetry,
- });
- }
- /**
- * @returns {string} path to cache dir
- */
- 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 {}
- const parent = path.dirname(dir);
- if (dir === parent) {
- 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 compiler
- * @returns {boolean} true when target is `web`, otherwise false
- */
- static isWebTarget(compiler) {
- if (compiler.platform && compiler.platform.web) {
- return compiler.platform.web;
- }
- // TODO improve for the next major version and keep only `webTargets` to fallback for old versions
- if (
- compiler.options.externalsPresets &&
- compiler.options.externalsPresets.web
- ) {
- return true;
- }
- if (
- compiler.options.resolve.conditionNames &&
- compiler.options.resolve.conditionNames.includes("browser")
- ) {
- return true;
- }
- const webTargets = [
- "web",
- "webworker",
- "electron-preload",
- "electron-renderer",
- "nwjs",
- "node-webkit",
- undefined,
- null,
- ];
- if (Array.isArray(compiler.options.target)) {
- return compiler.options.target.some((r) => webTargets.includes(r));
- }
- return webTargets.includes(/** @type {string} */ (compiler.options.target));
- }
- /**
- * @private
- * @param {Compiler} compiler compiler
- */
- addAdditionalEntries(compiler) {
- /**
- * @type {string[]}
- */
- const additionalEntries = [];
- const isWebTarget = Server.isWebTarget(compiler);
- // TODO maybe 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 = this.isTlsServer ? "wss:" : "ws:";
- }
- 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
- const isSockJSType = webSocketServer.type === "sockjs";
- const isWebSocketServerHostDefined =
- typeof webSocketServer.options.host !== "undefined";
- const isWebSocketServerPortDefined =
- typeof webSocketServer.options.port !== "undefined";
- if (
- isSockJSType &&
- (isWebSocketServerHostDefined || isWebSocketServerPortDefined)
- ) {
- this.logger.warn(
- "SockJS only supports client mode and does not support custom hostname and port options. Please consider using 'ws' if you need to customize these options.",
- );
- }
- // 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 (isWebSocketServerHostDefined && !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 (isWebSocketServerPortDefined && !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.progress !== "undefined") {
- searchParams.set("progress", String(client.progress));
- }
- if (typeof client.overlay !== "undefined") {
- const overlayString =
- typeof client.overlay === "boolean"
- ? String(client.overlay)
- : JSON.stringify({
- ...client.overlay,
- errors: encodeOverlaySettings(client.overlay.errors),
- warnings: encodeOverlaySettings(client.overlay.warnings),
- runtimeErrors: encodeOverlaySettings(
- client.overlay.runtimeErrors,
- ),
- });
- searchParams.set("overlay", overlayString);
- }
- if (typeof client.reconnect !== "undefined") {
- searchParams.set(
- "reconnect",
- typeof client.reconnect === "number"
- ? String(client.reconnect)
- : "10",
- );
- }
- if (typeof this.options.hot !== "undefined") {
- searchParams.set("hot", String(this.options.hot));
- }
- if (typeof this.options.liveReload !== "undefined") {
- searchParams.set("live-reload", String(this.options.liveReload));
- }
- webSocketURLStr = searchParams.toString();
- }
- additionalEntries.push(`${this.getClientEntry()}?${webSocketURLStr}`);
- }
- const clientHotEntry = this.getClientHotEntry();
- if (clientHotEntry) {
- additionalEntries.push(clientHotEntry);
- }
- const webpack = compiler.webpack || require("webpack");
- // use a hook to add entries if available
- for (const additionalEntry of additionalEntries) {
- new webpack.EntryPlugin(compiler.context, additionalEntry, {
- name: undefined,
- }).apply(compiler);
- }
- }
- /**
- * @private
- * @returns {Compiler["options"]} 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",
- 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();
- const compilerWatchOptions = compilerOptions.watchOptions;
- /**
- * @param {WatchOptions & { aggregateTimeout?: number, ignored?: WatchOptions["ignored"], poll?: number | boolean }} watchOptions watch options
- * @returns {WatchOptions} normalized watch options
- */
- 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 for static
- * @returns {NormalizedStatic} normalized options for static
- */
- const getStaticItem = (optionsForStatic) => {
- const getDefaultStaticOptions = () => ({
- 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,
- staticOptions:
- typeof optionsForStatic.staticOptions !== "undefined"
- ? { ...def.staticOptions, ...optionsForStatic.staticOptions }
- : def.staticOptions,
- publicPath:
- typeof optionsForStatic.publicPath !== "undefined"
- ? Array.isArray(optionsForStatic.publicPath)
- ? optionsForStatic.publicPath
- : [optionsForStatic.publicPath]
- : def.publicPath,
- serveIndex:
- // Check if 'serveIndex' property is defined in 'optionsForStatic'
- // If 'serveIndex' is a boolean and true, use default 'serveIndex'
- // If 'serveIndex' is an object, merge its properties with default 'serveIndex'
- // If 'serveIndex' is neither a boolean true nor an object, use it as-is
- // If 'serveIndex' is not defined in 'optionsForStatic', use default 'serveIndex'
- typeof optionsForStatic.serveIndex !== "undefined"
- ? typeof optionsForStatic.serveIndex === "boolean" &&
- optionsForStatic.serveIndex
- ? def.serveIndex
- : typeof optionsForStatic.serveIndex === "object"
- ? { ...def.serveIndex, ...optionsForStatic.serveIndex }
- : optionsForStatic.serveIndex
- : def.serveIndex,
- watch:
- typeof optionsForStatic.watch !== "undefined"
- ? 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;
- if (
- typeof options.server === "function" ||
- typeof options.server === "string"
- ) {
- options.server = {
- type: options.server,
- options: {},
- };
- } else {
- const serverOptions =
- /** @type {ServerConfiguration<A, S>} */
- (options.server || {});
- options.server = {
- type: serverOptions.type || "http",
- options: { ...serverOptions.options },
- };
- }
- const serverOptions = /** @type {ServerOptions} */ (options.server.options);
- if (
- options.server.type === "spdy" &&
- typeof serverOptions.spdy === "undefined"
- ) {
- serverOptions.spdy = { protocols: ["h2", "http/1.1"] };
- }
- if (
- options.server.type === "https" ||
- options.server.type === "http2" ||
- options.server.type === "spdy"
- ) {
- if (typeof serverOptions.requestCert === "undefined") {
- serverOptions.requestCert = false;
- }
- const httpsProperties =
- /** @type {Array<keyof ServerOptions>} */
- (["ca", "cert", "crl", "key", "pfx"]);
- for (const property_ of httpsProperties) {
- const property = /** @type {keyof ServerOptions} */ (property_);
- if (typeof serverOptions[property] === "undefined") {
- continue;
- }
- const value = serverOptions[property];
- /**
- * @param {string | Buffer | undefined} item file to read
- * @returns {string | Buffer | undefined} content of file
- */
- 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 {
- // Ignore error
- }
- // It is a file
- return stats ? fs.readFileSync(item) : item;
- }
- };
- /** @type {EXPECTED_ANY} */
- (serverOptions)[property] = Array.isArray(value)
- ? value.map((item) =>
- readFile(
- /** @type {string | Buffer | undefined} */
- (item),
- ),
- )
- : readFile(
- /** @type {string | Buffer | undefined} */
- (value),
- );
- }
- let fakeCert;
- if (!serverOptions.key || !serverOptions.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 = Date.now();
- // cert is more than 30 days old, kill it with fire
- if ((now - Number(certificateStat.ctime)) / certificateTtl > 30) {
- this.logger.info(
- "SSL certificate is more than 30 days old. Removing...",
- );
- await fs.promises.rm(certificatePath, { recursive: true });
- certificateExists = false;
- }
- }
- if (!certificateExists) {
- this.logger.info("Generating SSL certificate...");
- const selfsigned = require("selfsigned");
- const attributes = [{ name: "commonName", value: "localhost" }];
- const notBeforeDate = new Date();
- const notAfterDate = new Date();
- notAfterDate.setDate(notAfterDate.getDate() + 30);
- const pems = await selfsigned.generate(attributes, {
- algorithm: "sha256",
- keySize: 2048,
- notBeforeDate,
- notAfterDate,
- 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}`);
- }
- serverOptions.key ||= fakeCert;
- serverOptions.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;
- // https://github.com/webpack/webpack-dev-server/issues/1990
- const defaultOpenOptions = { wait: false };
- /**
- * @param {import("open").Options & { target?: string | string[] } & EXPECTED_ANY} target target
- * @returns {NormalizedOpen[]} normalized open options
- */
- 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) => ({
- 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 = [];
- for (const item of options.open) {
- if (typeof item === "string") {
- result.push({ target: item, options: defaultOpenOptions });
- continue;
- }
- result.push(...getOpenItemsFromObject(item));
- }
- /** @type {NormalizedOpen[]} */
- (options.open) = result;
- } else {
- /** @type {NormalizedOpen[]} */
- (options.open) = [...getOpenItemsFromObject(options.open)];
- }
- 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") {
- options.proxy = options.proxy.map((item) => {
- if (typeof item === "function") {
- return item;
- }
- /**
- * @param {"info" | "warn" | "error" | "debug" | "silent" | undefined | "none" | "log" | "verbose"} level level
- * @returns {"info" | "warn" | "error" | "debug" | "silent" | undefined} log level for proxy
- */
- 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} client transport
- */
- getClientTransport() {
- let clientImplementation;
- let clientImplementationFound = true;
- const isKnownWebSocketServerImplementation =
- this.options.webSocketServer &&
- typeof (
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type
- ) === "string" &&
- // @ts-expect-error
- (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 {
- 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);
- }
- /**
- * @template T
- * @private
- * @returns {T} server transport
- */
- 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 {
- implementation = require(
- /** @type {WebSocketServerConfiguration} */
- (this.options.webSocketServer).type,
- );
- } catch {
- 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;
- }
- /**
- * @returns {string}
- */
- getClientEntry() {
- return require.resolve("../client/index.js");
- }
- /**
- * @returns {string | void} client hot entry
- */
- getClientHotEntry() {
- if (this.options.hot === "only") {
- return require.resolve("webpack/hot/only-dev-server");
- } else if (this.options.hot) {
- return require.resolve("webpack/hot/dev-server");
- }
- }
- /**
- * @private
- * @returns {void}
- */
- setupProgressPlugin() {
- const { ProgressPlugin } =
- /** @type {MultiCompiler} */
- (this.compiler).compilers
- ? /** @type {MultiCompiler} */ (this.compiler).compilers[0].webpack
- : /** @type {Compiler} */ (this.compiler).webpack;
- new ProgressPlugin(
- /**
- * @param {number} percent percent
- * @param {string} msg message
- * @param {string} addInfo extra information
- * @param {string} pluginName plugin name
- */
- (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() {
- this.setupHooks();
- await this.setupApp();
- await this.createServer();
- if (this.options.webSocketServer) {
- const compilers =
- /** @type {MultiCompiler} */
- (this.compiler).compilers || [this.compiler];
- for (const compiler of compilers) {
- if (compiler.options.devServer === false) {
- continue;
- }
- this.addAdditionalEntries(compiler);
- const webpack = compiler.webpack || require("webpack");
- new webpack.ProvidePlugin({
- __webpack_dev_server_client__: this.getClientTransport(),
- }).apply(compiler);
- if (this.options.hot) {
- const HMRPluginExists = compiler.options.plugins.find(
- (plugin) =>
- plugin &&
- plugin.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.setupWatchFiles();
- this.setupWatchStaticFiles();
- this.setupMiddlewares();
- if (this.options.setupExitSignals) {
- const signals = ["SIGINT", "SIGTERM"];
- let needForceShutdown = false;
- for (const signal of signals) {
- // eslint-disable-next-line no-loop-func
- const listener = () => {
- if (needForceShutdown) {
- // eslint-disable-next-line n/no-process-exit
- 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(() => {
- // eslint-disable-next-line n/no-process-exit
- process.exit();
- });
- } else {
- // eslint-disable-next-line n/no-process-exit
- 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
- const webSocketProxies =
- /** @type {RequestHandler[]} */
- (this.webSocketProxies);
- for (const webSocketProxy of webSocketProxies) {
- /** @type {S} */
- (this.server).on(
- "upgrade",
- /** @type {RequestHandler & { upgrade: NonNullable<RequestHandler["upgrade"]> }} */
- (webSocketProxy).upgrade,
- );
- }
- }
- /**
- * @private
- * @returns {Promise<void>}
- */
- async setupApp() {
- /** @type {A | undefined} */
- this.app =
- /** @type {A} */
- (
- typeof this.options.app === "function"
- ? await this.options.app()
- : getExpress()()
- );
- }
- /**
- * @private
- * @param {Stats | MultiStats} statsObj stats
- * @returns {StatsCompilation} stats of compilation
- */
- getStats(statsObj) {
- const stats = Server.DEFAULT_STATS;
- const compilerOptions = this.getCompilerOptions();
- if (
- compilerOptions.stats &&
- /** @type {StatsOptions} */ (compilerOptions.stats).warningsFilter
- ) {
- stats.warningsFilter =
- /** @type {StatsOptions} */
- (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
- */
- (stats) => {
- if (this.webSocketServer) {
- this.sendStats(this.webSocketServer.clients, this.getStats(stats));
- }
- /**
- * @private
- * @type {Stats | MultiStats}
- */
- this.stats = stats;
- },
- );
- }
- /**
- * @private
- * @returns {void}
- */
- setupWatchStaticFiles() {
- const watchFiles = /** @type {NormalizedStatic[]} */ (this.options.static);
- if (watchFiles.length > 0) {
- for (const item of watchFiles) {
- if (item.watch) {
- this.watchFiles(item.directory, item.watch);
- }
- }
- }
- }
- /**
- * @private
- * @returns {void}
- */
- setupWatchFiles() {
- const watchFiles = /** @type {WatchFiles[]} */ (this.options.watchFiles);
- if (watchFiles.length > 0) {
- for (const item of watchFiles) {
- this.watchFiles(item.paths, item.options);
- }
- }
- }
- /**
- * @private
- * @returns {void}
- */
- setupMiddlewares() {
- /**
- * @type {Array<Middleware>}
- */
- let middlewares = [];
- // Register setup host header check for security
- middlewares.push({
- name: "host-header-check",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- const headers =
- /** @type {{ [key: string]: string | undefined }} */
- (req.headers);
- const headerName = headers[":authority"] ? ":authority" : "host";
- if (this.isValidHost(headers, headerName)) {
- next();
- return;
- }
- res.statusCode = 403;
- res.end("Invalid Host header");
- },
- });
- // Register setup cross origin request check for security
- middlewares.push({
- name: "cross-origin-header-check",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- const headers =
- /** @type {{ [key: string]: string | undefined }} */
- (req.headers);
- const headerName = headers[":authority"] ? ":authority" : "host";
- if (this.isValidHost(headers, headerName, false)) {
- next();
- return;
- }
- if (
- headers["sec-fetch-mode"] === "no-cors" &&
- headers["sec-fetch-site"] === "cross-site"
- ) {
- res.statusCode = 403;
- res.end("Cross-Origin request blocked");
- return;
- }
- next();
- },
- });
- const isHTTP2 =
- /** @type {ServerConfiguration<A, S>} */ (this.options.server).type ===
- "http2";
- if (isHTTP2) {
- // TODO patch for https://github.com/pillarjs/finalhandler/pull/45, need remove then will be resolved
- middlewares.push({
- name: "http2-status-message-patch",
- middleware:
- /** @type {NextHandleFunction} */
- (_req, res, next) => {
- Object.defineProperty(res, "statusMessage", {
- get() {
- return "";
- },
- set() {},
- });
- next();
- },
- });
- }
- // compress is placed last and uses unshift so that it will be the first middleware used
- if (this.options.compress && !isHTTP2) {
- const compression = require("compression");
- middlewares.push({ name: "compression", middleware: compression() });
- }
- if (typeof this.options.headers !== "undefined") {
- middlewares.push({
- name: "set-headers",
- middleware: this.setHeaders.bind(this),
- });
- }
- middlewares.push({
- name: "webpack-dev-middleware",
- middleware: /** @type {MiddlewareHandler} */ (this.middleware),
- });
- // Should be after `webpack-dev-middleware`, otherwise other middlewares might rewrite response
- middlewares.push({
- name: "webpack-dev-server-sockjs-bundle",
- path: "/__webpack_dev_server__/sockjs.bundle.js",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- if (req.method !== "GET" && req.method !== "HEAD") {
- next();
- return;
- }
- const clientPath = path.join(
- __dirname,
- "..",
- "client/modules/sockjs-client/index.js",
- );
- // Express send Etag and other headers by default, so let's keep them for compatibility reasons
- if (typeof res.sendFile === "function") {
- res.sendFile(clientPath);
- return;
- }
- let stats;
- try {
- // TODO implement `inputFileSystem.createReadStream` in webpack
- stats = fs.statSync(clientPath);
- } catch {
- next();
- return;
- }
- res.setHeader("Content-Type", "application/javascript; charset=UTF-8");
- res.setHeader("Content-Length", stats.size);
- if (req.method === "HEAD") {
- res.end();
- return;
- }
- fs.createReadStream(clientPath).pipe(res);
- },
- });
- middlewares.push({
- name: "webpack-dev-server-invalidate",
- path: "/webpack-dev-server/invalidate",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- if (req.method !== "GET" && req.method !== "HEAD") {
- next();
- return;
- }
- this.invalidate();
- res.end();
- },
- });
- middlewares.push({
- name: "webpack-dev-server-open-editor",
- path: "/webpack-dev-server/open-editor",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- if (req.method !== "GET" && req.method !== "HEAD") {
- next();
- return;
- }
- if (!req.url) {
- next();
- return;
- }
- const resolveUrl = new URL(req.url, `http://${req.headers.host}`);
- const params = new URLSearchParams(resolveUrl.search);
- const fileName = params.get("fileName");
- if (typeof fileName === "string") {
- const launchEditor = require("launch-editor");
- launchEditor(fileName);
- }
- res.end();
- },
- });
- middlewares.push({
- name: "webpack-dev-server-assets",
- path: "/webpack-dev-server",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- if (req.method !== "GET" && req.method !== "HEAD") {
- next();
- return;
- }
- if (!this.middleware) {
- next();
- return;
- }
- this.middleware.waitUntilValid((stats) => {
- res.setHeader("Content-Type", "text/html; charset=utf-8");
- // HEAD requests should not return body content
- if (req.method === "HEAD") {
- res.end();
- return;
- }
- res.write(
- '<!DOCTYPE html><html><head><meta charset="utf-8"/></head><body>',
- );
- /**
- * @type {StatsCompilation[]}
- */
- const statsForPrint =
- typeof (/** @type {MultiStats} */ (stats).stats) !== "undefined"
- ? /** @type {NonNullable<StatsCompilation["children"]>} */
- (/** @type {MultiStats} */ (stats).toJson().children)
- : [/** @type {Stats} */ (stats).toJson()];
- res.write("<h1>Assets Report:</h1>");
- for (const [index, item] of statsForPrint.entries()) {
- res.write("<div>");
- const name =
- 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;
- const assets =
- /** @type {NonNullable<StatsCompilation["assets"]>} */
- (item.assets);
- for (const asset of 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>");
- });
- },
- });
- if (this.options.proxy) {
- const { createProxyMiddleware } = require("http-proxy-middleware");
- /**
- * @param {ProxyConfigArrayItem} proxyConfig proxy config
- * @returns {RequestHandler | undefined} request handler
- */
- 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);
- }
- // TODO improve me after drop `bypass` to always generate error when configuration is bad
- if (!proxyConfig.bypass) {
- util.deprecate(
- () => {},
- `Invalid proxy configuration:\n\n${JSON.stringify(proxyConfig, null, 2)}\n\nThe use of proxy object notation as proxy routes has been removed.\nPlease use the 'router' or 'context' options. Read more at https://github.com/chimurai/http-proxy-middleware/tree/v2.0.6#http-proxy-middleware-options`,
- "DEP_WEBPACK_DEV_SERVER_PROXY_ROUTES_ARGUMENT",
- )();
- }
- };
- /**
- * @example
- * Assume a proxy configuration specified as:
- * proxy: [
- * {
- * context: "value",
- * ...options,
- * },
- * // or:
- * function() {
- * return {
- * context: "context",
- * ...options,
- * };
- * }
- * ]
- */
- for (const proxyConfigOrCallback of this.options.proxy) {
- /**
- * @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 request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {Promise<void>}
- */
- const handler = async (req, res, next) => {
- if (typeof proxyConfigOrCallback === "function") {
- const newProxyConfig = proxyConfigOrCallback(req, res, next);
- if (newProxyConfig !== proxyConfig) {
- proxyConfig = newProxyConfig;
- const socket = req.socket || req.connection;
- // @ts-expect-error
- const server = socket ? socket.server : null;
- if (server) {
- server.removeAllListeners("close");
- }
- 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";
- if (isByPassFuncDefined) {
- util.deprecate(
- () => {},
- "Using the 'bypass' option is deprecated. Please use the 'router' or 'context' options. Read more at https://github.com/chimurai/http-proxy-middleware/tree/v2.0.6#http-proxy-middleware-options",
- "DEP_WEBPACK_DEV_SERVER_PROXY_BYPASS_ARGUMENT",
- )();
- }
- const bypassUrl = isByPassFuncDefined
- ? await /** @type {ByPass} */ (proxyConfig.bypass)(
- req,
- res,
- proxyConfig,
- )
- : null;
- if (typeof bypassUrl === "boolean") {
- // skip the proxy
- res.statusCode = 404;
- req.url = "";
- 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 error
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {Promise<void>} nothing
- */
- (error, req, res, next) => handler(req, res, next),
- });
- }
- middlewares.push({
- name: "webpack-dev-middleware",
- middleware: /** @type {MiddlewareHandler} */ (this.middleware),
- });
- }
- const staticOptions =
- /** @type {NormalizedStatic[]} */
- (this.options.static);
- if (staticOptions.length > 0) {
- for (const staticOption of staticOptions) {
- for (const publicPath of staticOption.publicPath) {
- middlewares.push({
- name: "express-static",
- path: publicPath,
- middleware: getExpress().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-expect-error
- 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 {MiddlewareHandler} */ (this.middleware),
- });
- if (staticOptions.length > 0) {
- for (const staticOption of staticOptions) {
- for (const publicPath of staticOption.publicPath) {
- middlewares.push({
- name: "express-static",
- path: publicPath,
- middleware: getExpress().static(
- staticOption.directory,
- staticOption.staticOptions,
- ),
- });
- }
- }
- }
- }
- if (staticOptions.length > 0) {
- const serveIndex = require("serve-index");
- for (const staticOption of staticOptions) {
- for (const publicPath of staticOption.publicPath) {
- if (staticOption.serveIndex) {
- middlewares.push({
- name: "serve-index",
- path: publicPath,
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @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);
- },
- });
- }
- }
- }
- }
- // Register this middleware always as the last one so that it's only used as a
- // fallback when no other middleware responses.
- middlewares.push({
- name: "options-middleware",
- /**
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- * @returns {void}
- */
- middleware: (req, res, next) => {
- if (req.method === "OPTIONS") {
- res.statusCode = 204;
- res.setHeader("Content-Length", "0");
- res.end();
- return;
- }
- next();
- },
- });
- if (typeof this.options.setupMiddlewares === "function") {
- middlewares = this.options.setupMiddlewares(middlewares, this);
- }
- // Lazy init webpack dev middleware
- const lazyInitDevMiddleware = () => {
- if (!this.middleware) {
- const webpackDevMiddleware = require("webpack-dev-middleware");
- // middleware for serving webpack bundle
- /** @type {import("webpack-dev-middleware").API<Request, Response>} */
- this.middleware = webpackDevMiddleware(
- this.compiler,
- this.options.devMiddleware,
- );
- }
- return this.middleware;
- };
- for (const i of middlewares) {
- if (i.name === "webpack-dev-middleware") {
- const item = /** @type {MiddlewareObject} */ (i);
- if (typeof item.middleware === "undefined") {
- item.middleware = lazyInitDevMiddleware();
- }
- }
- }
- for (const middleware of middlewares) {
- if (typeof middleware === "function") {
- /** @type {A} */
- (this.app).use(
- /** @type {NextHandleFunction | HandleFunction} */
- (middleware),
- );
- } else if (typeof middleware.path !== "undefined") {
- /** @type {A} */
- (this.app).use(
- middleware.path,
- /** @type {SimpleHandleFunction | NextHandleFunction} */
- (middleware.middleware),
- );
- } else {
- /** @type {A} */
- (this.app).use(
- /** @type {NextHandleFunction | HandleFunction} */
- (middleware.middleware),
- );
- }
- }
- }
- /**
- * @private
- * @returns {Promise<void>}
- */
- async createServer() {
- const { type, options } =
- /** @type {ServerConfiguration<A, S>} */
- (this.options.server);
- if (typeof type === "function") {
- /** @type {S | undefined} */
- this.server = await type(
- /** @type {ServerOptions} */
- (options),
- /** @type {A} */
- (this.app),
- );
- } else {
- const serverType = require(/** @type {string} */ (type));
- /** @type {S | undefined} */
- this.server =
- type === "http2"
- ? serverType.createSecureServer(
- { ...options, allowHTTP1: true },
- this.app,
- )
- : serverType.createServer(options, this.app);
- }
- this.isTlsServer =
- typeof (
- /** @type {import("tls").Server} */ (this.server).setSecureContext
- ) !== "undefined";
- /** @type {S} */
- (this.server).on(
- "connection",
- /**
- * @param {Socket} socket connected 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 {S} */
- (this.server).on(
- "error",
- /**
- * @param {Error} error error
- */
- (error) => {
- throw error;
- },
- );
- }
- /**
- * @private
- * @returns {void}
- */
- createWebSocketServer() {
- /** @type {WebSocketServerImplementation | undefined | null} */
- this.webSocketServer = new (this.getServerTransport())(this);
- /** @type {WebSocketServerImplementation} */
- (this.webSocketServer).implementation.on(
- "connection",
- /**
- * @param {ClientConnection} client client
- * @param {IncomingMessage} request request
- */
- (client, request) => {
- /** @type {{ [key: string]: string | undefined } | undefined} */
- const headers =
- typeof request !== "undefined"
- ? /** @type {{ [key: string]: string | undefined }} */
- (request.headers)
- : typeof (
- /** @type {import("sockjs").Connection} */ (client).headers
- ) !== "undefined"
- ? /** @type {import("sockjs").Connection} */ (client).headers
- : undefined;
- if (!headers) {
- this.logger.warn(
- 'webSocketServer implementation must pass headers for the "connection" event',
- );
- }
- if (
- !headers ||
- !this.isValidHost(headers, "host") ||
- !this.isValidHost(headers, "origin") ||
- !this.isSameOrigin(headers)
- ) {
- 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
- ) {
- const overlayConfig =
- /** @type {ClientConfiguration} */
- (this.options.client).overlay;
- this.sendMessage(
- [client],
- "overlay",
- typeof overlayConfig === "object"
- ? {
- ...overlayConfig,
- errors:
- overlayConfig.errors &&
- encodeOverlaySettings(overlayConfig.errors),
- warnings:
- overlayConfig.warnings &&
- encodeOverlaySettings(overlayConfig.warnings),
- runtimeErrors:
- overlayConfig.runtimeErrors &&
- encodeOverlaySettings(overlayConfig.runtimeErrors),
- }
- : overlayConfig,
- );
- }
- if (!this.stats) {
- return;
- }
- this.sendStats([client], this.getStats(this.stats), true);
- },
- );
- }
- /**
- * @private
- * @param {string} defaultOpenTarget default open target
- * @returns {Promise<void>}
- */
- async openBrowser(defaultOpenTarget) {
- const open = (await import("open")).default;
- 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-name".`,
- );
- });
- }),
- );
- }
- /**
- * @private
- * @returns {void}
- */
- runBonjour() {
- const { Bonjour } = require("bonjour-service");
- const type = this.isTlsServer ? "https" : "http";
- /**
- * @private
- * @type {Bonjour | undefined}
- */
- this.bonjour = new Bonjour();
- this.bonjour.publish({
- name: `Webpack Dev Server ${os.hostname()}:${this.options.port}`,
- port: /** @type {number} */ (this.options.port),
- type,
- subtypes: ["webpack"],
- .../** @type {Partial<BonjourOptions>} */ (this.options.bonjour),
- });
- }
- /**
- * @private
- * @param {() => void} callback callback
- * @returns {void}
- */
- stopBonjour(callback = () => {}) {
- /** @type {Bonjour} */
- (this.bonjour).unpublishAll(() => {
- /** @type {Bonjour} */
- (this.bonjour).destroy();
- if (callback) {
- callback();
- }
- });
- }
- /**
- * @private
- * @returns {Promise<void>}
- */
- async logStatus() {
- const { cyan, isColorSupported, red } = require("colorette");
- /**
- * @param {Compiler["options"]} compilerOptions compiler options
- * @returns {boolean} value of the color option
- */
- 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 need to use color?
- * @param {string} msg message
- * @returns {string} message with color
- */
- info(useColor, msg) {
- if (useColor) {
- return cyan(msg);
- }
- return msg;
- },
- /**
- * @param {boolean} useColor need to use color?
- * @param {string} msg message
- * @returns {string} message with colors
- */
- error(useColor, msg) {
- if (useColor) {
- return red(msg);
- }
- return msg;
- },
- };
- const useColor = getColorsOption(this.getCompilerOptions());
- const server = /** @type {S} */ (this.server);
- if (this.options.ipc) {
- this.logger.info(`Project is running at: "${server.address()}"`);
- } else {
- const protocol = this.isTlsServer ? "https" : "http";
- const { address, port } =
- /** @type {import("net").AddressInfo} */
- (server.address());
- /**
- * @param {string} newHostname new hostname
- * @returns {string} prettified URL
- */
- const prettyPrintURL = (newHostname) =>
- url.format({ protocol, hostname: newHostname, port, pathname: "/" });
- let host;
- 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 {
- // Ignore
- }
- if (!isIP) {
- host = prettyPrintURL(this.options.host);
- }
- }
- }
- const parsedIP = ipaddr.parse(address);
- if (parsedIP.range() === "unspecified") {
- localhost = prettyPrintURL("localhost");
- loopbackIPv6 = prettyPrintURL("::1");
- const networkIPv4 = Server.findIp("v4", false);
- if (networkIPv4) {
- networkUrlIPv4 = prettyPrintURL(networkIPv4);
- }
- const networkIPv6 = Server.findIp("v6", false);
- 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 (host) {
- this.logger.info(`Server: ${colors.info(useColor, host)}`);
- }
- 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 ||
- this.options.host === "0.0.0.0" ||
- this.options.host === "::"
- ? "localhost"
- : this.options.host,
- );
- await 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 || this.isTlsServer ? "https" : "http";
- this.logger.info(
- `Broadcasting "${bonjourProtocol}" with subtype of "webpack" via ZeroConf DNS (Bonjour)`,
- );
- }
- }
- /**
- * @private
- * @param {Request} req request
- * @param {Response} res response
- * @param {NextFunction} next next function
- */
- setHeaders(req, res, next) {
- let { headers } = this.options;
- if (headers) {
- if (typeof headers === "function") {
- headers = headers(
- req,
- res,
- this.middleware ? this.middleware.context : undefined,
- );
- }
- /**
- * @type {{ key: string, value: string }[]}
- */
- const allHeaders = [];
- if (!Array.isArray(headers)) {
- for (const name in headers) {
- allHeaders.push({
- key: name,
- value: /** @type {string} */ (headers[name]),
- });
- }
- headers = allHeaders;
- }
- for (const { key, value } of headers) {
- res.setHeader(key, value);
- }
- }
- next();
- }
- /**
- * @private
- * @param {string} value value
- * @returns {boolean} true when host allowed, otherwise false
- */
- isHostAllowed(value) {
- const { allowedHosts } = this.options;
- // allow user to opt out of this security check, at their own risk
- // by explicitly enabling allowedHosts
- if (allowedHosts === "all") {
- return true;
- }
- // always allow localhost host, for convenience
- // allow if value is in allowedHosts
- if (Array.isArray(allowedHosts) && allowedHosts.length > 0) {
- for (const allowedHost of allowedHosts) {
- if (allowedHost === value) {
- return true;
- }
- // support "." as a subdomain wildcard
- // e.g. ".example.com" will allow "example.com", "www.example.com", "subdomain.example.com", etc
- if (
- allowedHost.startsWith(".") && // "example.com" (value === allowedHost.substring(1))
- // "*.example.com" (value.endsWith(allowedHost))
- (value === allowedHost.slice(1) ||
- /** @type {string} */
- (value).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 === value
- );
- }
- return false;
- }
- /**
- * @private
- * @param {{ [key: string]: string | undefined }} headers headers
- * @param {string} headerToCheck header to check
- * @param {boolean} validateHost need to validate host
- * @returns {boolean} true when host is valid, otherwise false
- */
- isValidHost(headers, headerToCheck, validateHost = true) {
- if (this.options.allowedHosts === "all") {
- return true;
- }
- // get the Host header and extract hostname
- // we don't care about port not matching
- const header = headers[headerToCheck];
- if (!header) {
- return false;
- }
- if (DEFAULT_ALLOWED_PROTOCOLS.test(header)) {
- return true;
- }
- // use the node url-parser to retrieve the hostname from the host-header.
- // TODO resolve me in the next major release
- // eslint-disable-next-line n/no-deprecated-api
- const { hostname } = url.parse(
- // if header doesn't have scheme, add // for parsing.
- /^(.+:)?\/\//.test(header) ? header : `//${header}`,
- false,
- true,
- );
- if (hostname === null) {
- return false;
- }
- if (this.isHostAllowed(hostname)) {
- return true;
- }
- // always allow requests with explicit IPv4 or IPv6-address.
- // A note on IPv6 addresses:
- // header 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.
- // For convenience, always allow localhost (hostname === 'localhost')
- // and its subdomains (hostname.endsWith(".localhost")).
- // allow hostname of listening address (hostname === this.options.host)
- const isValidHostname = validateHost
- ? ipaddr.IPv4.isValid(hostname) ||
- ipaddr.IPv6.isValid(hostname) ||
- hostname === "localhost" ||
- hostname.endsWith(".localhost") ||
- hostname === this.options.host
- : false;
- return isValidHostname;
- }
- /**
- * @private
- * @param {{ [key: string]: string | undefined }} headers headers
- * @returns {boolean} true when is same origin, otherwise false
- */
- isSameOrigin(headers) {
- if (this.options.allowedHosts === "all") {
- return true;
- }
- const originHeader = headers.origin;
- if (!originHeader) {
- return this.options.allowedHosts === "all";
- }
- if (DEFAULT_ALLOWED_PROTOCOLS.test(originHeader)) {
- return true;
- }
- // TODO resolve me in the next major release
- // eslint-disable-next-line n/no-deprecated-api
- const origin = url.parse(originHeader, false, true).hostname;
- if (origin === null) {
- return false;
- }
- if (this.isHostAllowed(origin)) {
- return true;
- }
- const hostHeader = headers.host;
- if (!hostHeader) {
- return this.options.allowedHosts === "all";
- }
- if (DEFAULT_ALLOWED_PROTOCOLS.test(hostHeader)) {
- return true;
- }
- // eslint-disable-next-line n/no-deprecated-api
- const host = url.parse(
- // if hostHeader doesn't have scheme, add // for parsing.
- /^(.+:)?\/\//.test(hostHeader) ? hostHeader : `//${hostHeader}`,
- false,
- true,
- ).hostname;
- if (host === null) {
- return false;
- }
- if (this.isHostAllowed(host)) {
- return true;
- }
- return origin === host;
- }
- /**
- * @param {ClientConnection[]} clients clients
- * @param {string} type type
- * @param {EXPECTED_ANY=} data data
- * @param {EXPECTED_ANY=} params params
- */
- 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 }));
- }
- }
- }
- // Send stats to a socket or multiple sockets
- /**
- * @private
- * @param {ClientConnection[]} clients clients
- * @param {StatsCompilation} stats stats
- * @param {boolean=} force 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 watch path
- * @param {WatchOptions=} watchOptions watch options
- */
- 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 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("node:net");
- const socket = new net.Socket();
- socket.on(
- "error",
- /**
- * @param {Error & { code?: string }} error 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),
- this.options.host,
- );
- }
- 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 {S} */
- (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();
- }
- await this.logStatus();
- if (typeof this.options.onListening === "function") {
- this.options.onListening(this);
- }
- }
- /**
- * @param {((err?: Error) => void)=} callback 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 {S} */
- (this.server).close(() => {
- this.server = undefined;
- 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 = undefined;
- }
- }
- // 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 callback
- */
- stopCallback(callback = () => {}) {
- this.stop()
- .then(() => callback(), callback)
- .catch(callback);
- }
- }
- module.exports = Server;
|