ContextModule.js 32 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const { OriginalSource, RawSource } = require("webpack-sources");
  7. const AsyncDependenciesBlock = require("./AsyncDependenciesBlock");
  8. const { makeWebpackError } = require("./HookWebpackError");
  9. const Module = require("./Module");
  10. const RuntimeGlobals = require("./RuntimeGlobals");
  11. const Template = require("./Template");
  12. const WebpackError = require("./WebpackError");
  13. const {
  14. compareLocations,
  15. concatComparators,
  16. compareSelect,
  17. keepOriginalOrder,
  18. compareModulesById
  19. } = require("./util/comparators");
  20. const { contextify, parseResource } = require("./util/identifier");
  21. const makeSerializable = require("./util/makeSerializable");
  22. /** @typedef {import("webpack-sources").Source} Source */
  23. /** @typedef {import("../declarations/WebpackOptions").WebpackOptionsNormalized} WebpackOptions */
  24. /** @typedef {import("./ChunkGraph")} ChunkGraph */
  25. /** @typedef {import("./ChunkGroup").RawChunkGroupOptions} RawChunkGroupOptions */
  26. /** @typedef {import("./Compilation")} Compilation */
  27. /** @typedef {import("./DependencyTemplates")} DependencyTemplates */
  28. /** @typedef {import("./Module").BuildMeta} BuildMeta */
  29. /** @typedef {import("./Module").CodeGenerationContext} CodeGenerationContext */
  30. /** @typedef {import("./Module").CodeGenerationResult} CodeGenerationResult */
  31. /** @typedef {import("./Module").LibIdentOptions} LibIdentOptions */
  32. /** @typedef {import("./Module").NeedBuildContext} NeedBuildContext */
  33. /** @typedef {import("./ModuleGraph")} ModuleGraph */
  34. /** @typedef {import("./RequestShortener")} RequestShortener */
  35. /** @typedef {import("./ResolverFactory").ResolverWithOptions} ResolverWithOptions */
  36. /** @typedef {import("./RuntimeTemplate")} RuntimeTemplate */
  37. /** @typedef {import("./dependencies/ContextElementDependency")} ContextElementDependency */
  38. /** @template T @typedef {import("./util/LazySet")<T>} LazySet<T> */
  39. /** @typedef {import("./util/fs").InputFileSystem} InputFileSystem */
  40. /** @typedef {"sync" | "eager" | "weak" | "async-weak" | "lazy" | "lazy-once"} ContextMode Context mode */
  41. /**
  42. * @typedef {Object} ContextOptions
  43. * @property {ContextMode} mode
  44. * @property {boolean} recursive
  45. * @property {RegExp} regExp
  46. * @property {"strict"|boolean=} namespaceObject
  47. * @property {string=} addon
  48. * @property {string=} chunkName
  49. * @property {RegExp=} include
  50. * @property {RegExp=} exclude
  51. * @property {RawChunkGroupOptions=} groupOptions
  52. * @property {string=} typePrefix
  53. * @property {string=} category
  54. * @property {string[][]=} referencedExports exports referenced from modules (won't be mangled)
  55. */
  56. /**
  57. * @typedef {Object} ContextModuleOptionsExtras
  58. * @property {string} resource
  59. * @property {string=} resourceQuery
  60. * @property {string=} resourceFragment
  61. * @property {TODO} resolveOptions
  62. */
  63. /** @typedef {ContextOptions & ContextModuleOptionsExtras} ContextModuleOptions */
  64. /**
  65. * @callback ResolveDependenciesCallback
  66. * @param {Error=} err
  67. * @param {ContextElementDependency[]=} dependencies
  68. */
  69. /**
  70. * @callback ResolveDependencies
  71. * @param {InputFileSystem} fs
  72. * @param {ContextModuleOptions} options
  73. * @param {ResolveDependenciesCallback} callback
  74. */
  75. const SNAPSHOT_OPTIONS = { timestamp: true };
  76. const TYPES = new Set(["javascript"]);
  77. class ContextModule extends Module {
  78. /**
  79. * @param {ResolveDependencies} resolveDependencies function to get dependencies in this context
  80. * @param {ContextModuleOptions} options options object
  81. */
  82. constructor(resolveDependencies, options) {
  83. const parsed = parseResource(options ? options.resource : "");
  84. const resource = parsed.path;
  85. const resourceQuery = (options && options.resourceQuery) || parsed.query;
  86. const resourceFragment =
  87. (options && options.resourceFragment) || parsed.fragment;
  88. super("javascript/dynamic", resource);
  89. // Info from Factory
  90. this.resolveDependencies = resolveDependencies;
  91. /** @type {ContextModuleOptions} */
  92. this.options = {
  93. ...options,
  94. resource,
  95. resourceQuery,
  96. resourceFragment
  97. };
  98. if (options && options.resolveOptions !== undefined) {
  99. this.resolveOptions = options.resolveOptions;
  100. }
  101. if (options && typeof options.mode !== "string") {
  102. throw new Error("options.mode is a required option");
  103. }
  104. this._identifier = this._createIdentifier();
  105. this._forceBuild = true;
  106. }
  107. /**
  108. * @returns {Set<string>} types available (do not mutate)
  109. */
  110. getSourceTypes() {
  111. return TYPES;
  112. }
  113. /**
  114. * Assuming this module is in the cache. Update the (cached) module with
  115. * the fresh module from the factory. Usually updates internal references
  116. * and properties.
  117. * @param {Module} module fresh module
  118. * @returns {void}
  119. */
  120. updateCacheModule(module) {
  121. const m = /** @type {ContextModule} */ (module);
  122. this.resolveDependencies = m.resolveDependencies;
  123. this.options = m.options;
  124. }
  125. /**
  126. * Assuming this module is in the cache. Remove internal references to allow freeing some memory.
  127. */
  128. cleanupForCache() {
  129. super.cleanupForCache();
  130. this.resolveDependencies = undefined;
  131. }
  132. prettyRegExp(regexString) {
  133. // remove the "/" at the front and the beginning
  134. // "/foo/" -> "foo"
  135. return regexString
  136. .substring(1, regexString.length - 1)
  137. .replace(/!/g, "%21");
  138. }
  139. _createIdentifier() {
  140. let identifier = this.context;
  141. if (this.options.resourceQuery) {
  142. identifier += `|${this.options.resourceQuery}`;
  143. }
  144. if (this.options.resourceFragment) {
  145. identifier += `|${this.options.resourceFragment}`;
  146. }
  147. if (this.options.mode) {
  148. identifier += `|${this.options.mode}`;
  149. }
  150. if (!this.options.recursive) {
  151. identifier += "|nonrecursive";
  152. }
  153. if (this.options.addon) {
  154. identifier += `|${this.options.addon}`;
  155. }
  156. if (this.options.regExp) {
  157. identifier += `|${this.options.regExp}`;
  158. }
  159. if (this.options.include) {
  160. identifier += `|include: ${this.options.include}`;
  161. }
  162. if (this.options.exclude) {
  163. identifier += `|exclude: ${this.options.exclude}`;
  164. }
  165. if (this.options.referencedExports) {
  166. identifier += `|referencedExports: ${JSON.stringify(
  167. this.options.referencedExports
  168. )}`;
  169. }
  170. if (this.options.chunkName) {
  171. identifier += `|chunkName: ${this.options.chunkName}`;
  172. }
  173. if (this.options.groupOptions) {
  174. identifier += `|groupOptions: ${JSON.stringify(
  175. this.options.groupOptions
  176. )}`;
  177. }
  178. if (this.options.namespaceObject === "strict") {
  179. identifier += "|strict namespace object";
  180. } else if (this.options.namespaceObject) {
  181. identifier += "|namespace object";
  182. }
  183. return identifier;
  184. }
  185. /**
  186. * @returns {string} a unique identifier of the module
  187. */
  188. identifier() {
  189. return this._identifier;
  190. }
  191. /**
  192. * @param {RequestShortener} requestShortener the request shortener
  193. * @returns {string} a user readable identifier of the module
  194. */
  195. readableIdentifier(requestShortener) {
  196. let identifier = requestShortener.shorten(this.context) + "/";
  197. if (this.options.resourceQuery) {
  198. identifier += ` ${this.options.resourceQuery}`;
  199. }
  200. if (this.options.mode) {
  201. identifier += ` ${this.options.mode}`;
  202. }
  203. if (!this.options.recursive) {
  204. identifier += " nonrecursive";
  205. }
  206. if (this.options.addon) {
  207. identifier += ` ${requestShortener.shorten(this.options.addon)}`;
  208. }
  209. if (this.options.regExp) {
  210. identifier += ` ${this.prettyRegExp(this.options.regExp + "")}`;
  211. }
  212. if (this.options.include) {
  213. identifier += ` include: ${this.prettyRegExp(this.options.include + "")}`;
  214. }
  215. if (this.options.exclude) {
  216. identifier += ` exclude: ${this.prettyRegExp(this.options.exclude + "")}`;
  217. }
  218. if (this.options.referencedExports) {
  219. identifier += ` referencedExports: ${this.options.referencedExports
  220. .map(e => e.join("."))
  221. .join(", ")}`;
  222. }
  223. if (this.options.chunkName) {
  224. identifier += ` chunkName: ${this.options.chunkName}`;
  225. }
  226. if (this.options.groupOptions) {
  227. const groupOptions = this.options.groupOptions;
  228. for (const key of Object.keys(groupOptions)) {
  229. identifier += ` ${key}: ${groupOptions[key]}`;
  230. }
  231. }
  232. if (this.options.namespaceObject === "strict") {
  233. identifier += " strict namespace object";
  234. } else if (this.options.namespaceObject) {
  235. identifier += " namespace object";
  236. }
  237. return identifier;
  238. }
  239. /**
  240. * @param {LibIdentOptions} options options
  241. * @returns {string | null} an identifier for library inclusion
  242. */
  243. libIdent(options) {
  244. let identifier = contextify(
  245. options.context,
  246. this.context,
  247. options.associatedObjectForCache
  248. );
  249. if (this.options.mode) {
  250. identifier += ` ${this.options.mode}`;
  251. }
  252. if (this.options.recursive) {
  253. identifier += " recursive";
  254. }
  255. if (this.options.addon) {
  256. identifier += ` ${contextify(
  257. options.context,
  258. this.options.addon,
  259. options.associatedObjectForCache
  260. )}`;
  261. }
  262. if (this.options.regExp) {
  263. identifier += ` ${this.prettyRegExp(this.options.regExp + "")}`;
  264. }
  265. if (this.options.include) {
  266. identifier += ` include: ${this.prettyRegExp(this.options.include + "")}`;
  267. }
  268. if (this.options.exclude) {
  269. identifier += ` exclude: ${this.prettyRegExp(this.options.exclude + "")}`;
  270. }
  271. if (this.options.referencedExports) {
  272. identifier += ` referencedExports: ${this.options.referencedExports
  273. .map(e => e.join("."))
  274. .join(", ")}`;
  275. }
  276. return identifier;
  277. }
  278. /**
  279. * @returns {void}
  280. */
  281. invalidateBuild() {
  282. this._forceBuild = true;
  283. }
  284. /**
  285. * @param {NeedBuildContext} context context info
  286. * @param {function(WebpackError=, boolean=): void} callback callback function, returns true, if the module needs a rebuild
  287. * @returns {void}
  288. */
  289. needBuild({ fileSystemInfo }, callback) {
  290. // build if enforced
  291. if (this._forceBuild) return callback(null, true);
  292. // always build when we have no snapshot
  293. if (!this.buildInfo.snapshot) return callback(null, true);
  294. fileSystemInfo.checkSnapshotValid(this.buildInfo.snapshot, (err, valid) => {
  295. callback(err, !valid);
  296. });
  297. }
  298. /**
  299. * @param {WebpackOptions} options webpack options
  300. * @param {Compilation} compilation the compilation
  301. * @param {ResolverWithOptions} resolver the resolver
  302. * @param {InputFileSystem} fs the file system
  303. * @param {function(WebpackError=): void} callback callback function
  304. * @returns {void}
  305. */
  306. build(options, compilation, resolver, fs, callback) {
  307. this._forceBuild = false;
  308. /** @type {BuildMeta} */
  309. this.buildMeta = {
  310. exportsType: "default",
  311. defaultObject: "redirect-warn"
  312. };
  313. this.buildInfo = {
  314. snapshot: undefined
  315. };
  316. this.dependencies.length = 0;
  317. this.blocks.length = 0;
  318. const startTime = Date.now();
  319. this.resolveDependencies(fs, this.options, (err, dependencies) => {
  320. if (err) {
  321. return callback(
  322. makeWebpackError(err, "ContextModule.resolveDependencies")
  323. );
  324. }
  325. // abort if something failed
  326. // this will create an empty context
  327. if (!dependencies) {
  328. callback();
  329. return;
  330. }
  331. // enhance dependencies with meta info
  332. for (const dep of dependencies) {
  333. dep.loc = {
  334. name: dep.userRequest
  335. };
  336. dep.request = this.options.addon + dep.request;
  337. }
  338. dependencies.sort(
  339. concatComparators(
  340. compareSelect(a => a.loc, compareLocations),
  341. keepOriginalOrder(this.dependencies)
  342. )
  343. );
  344. if (this.options.mode === "sync" || this.options.mode === "eager") {
  345. // if we have an sync or eager context
  346. // just add all dependencies and continue
  347. this.dependencies = dependencies;
  348. } else if (this.options.mode === "lazy-once") {
  349. // for the lazy-once mode create a new async dependency block
  350. // and add that block to this context
  351. if (dependencies.length > 0) {
  352. const block = new AsyncDependenciesBlock({
  353. ...this.options.groupOptions,
  354. name: this.options.chunkName
  355. });
  356. for (const dep of dependencies) {
  357. block.addDependency(dep);
  358. }
  359. this.addBlock(block);
  360. }
  361. } else if (
  362. this.options.mode === "weak" ||
  363. this.options.mode === "async-weak"
  364. ) {
  365. // we mark all dependencies as weak
  366. for (const dep of dependencies) {
  367. dep.weak = true;
  368. }
  369. this.dependencies = dependencies;
  370. } else if (this.options.mode === "lazy") {
  371. // if we are lazy create a new async dependency block per dependency
  372. // and add all blocks to this context
  373. let index = 0;
  374. for (const dep of dependencies) {
  375. let chunkName = this.options.chunkName;
  376. if (chunkName) {
  377. if (!/\[(index|request)\]/.test(chunkName)) {
  378. chunkName += "[index]";
  379. }
  380. chunkName = chunkName.replace(/\[index\]/g, `${index++}`);
  381. chunkName = chunkName.replace(
  382. /\[request\]/g,
  383. Template.toPath(dep.userRequest)
  384. );
  385. }
  386. const block = new AsyncDependenciesBlock(
  387. {
  388. ...this.options.groupOptions,
  389. name: chunkName
  390. },
  391. dep.loc,
  392. dep.userRequest
  393. );
  394. block.addDependency(dep);
  395. this.addBlock(block);
  396. }
  397. } else {
  398. callback(
  399. new WebpackError(`Unsupported mode "${this.options.mode}" in context`)
  400. );
  401. return;
  402. }
  403. compilation.fileSystemInfo.createSnapshot(
  404. startTime,
  405. null,
  406. [this.context],
  407. null,
  408. SNAPSHOT_OPTIONS,
  409. (err, snapshot) => {
  410. if (err) return callback(err);
  411. this.buildInfo.snapshot = snapshot;
  412. callback();
  413. }
  414. );
  415. });
  416. }
  417. /**
  418. * @param {LazySet<string>} fileDependencies set where file dependencies are added to
  419. * @param {LazySet<string>} contextDependencies set where context dependencies are added to
  420. * @param {LazySet<string>} missingDependencies set where missing dependencies are added to
  421. * @param {LazySet<string>} buildDependencies set where build dependencies are added to
  422. */
  423. addCacheDependencies(
  424. fileDependencies,
  425. contextDependencies,
  426. missingDependencies,
  427. buildDependencies
  428. ) {
  429. contextDependencies.add(this.context);
  430. }
  431. /**
  432. * @param {ContextElementDependency[]} dependencies all dependencies
  433. * @param {ChunkGraph} chunkGraph chunk graph
  434. * @returns {TODO} TODO
  435. */
  436. getUserRequestMap(dependencies, chunkGraph) {
  437. const moduleGraph = chunkGraph.moduleGraph;
  438. // if we filter first we get a new array
  439. // therefore we don't need to create a clone of dependencies explicitly
  440. // therefore the order of this is !important!
  441. const sortedDependencies = dependencies
  442. .filter(dependency => moduleGraph.getModule(dependency))
  443. .sort((a, b) => {
  444. if (a.userRequest === b.userRequest) {
  445. return 0;
  446. }
  447. return a.userRequest < b.userRequest ? -1 : 1;
  448. });
  449. const map = Object.create(null);
  450. for (const dep of sortedDependencies) {
  451. const module = moduleGraph.getModule(dep);
  452. map[dep.userRequest] = chunkGraph.getModuleId(module);
  453. }
  454. return map;
  455. }
  456. /**
  457. * @param {ContextElementDependency[]} dependencies all dependencies
  458. * @param {ChunkGraph} chunkGraph chunk graph
  459. * @returns {TODO} TODO
  460. */
  461. getFakeMap(dependencies, chunkGraph) {
  462. if (!this.options.namespaceObject) {
  463. return 9;
  464. }
  465. const moduleGraph = chunkGraph.moduleGraph;
  466. // bitfield
  467. let hasType = 0;
  468. const comparator = compareModulesById(chunkGraph);
  469. // if we filter first we get a new array
  470. // therefore we don't need to create a clone of dependencies explicitly
  471. // therefore the order of this is !important!
  472. const sortedModules = dependencies
  473. .map(dependency => moduleGraph.getModule(dependency))
  474. .filter(Boolean)
  475. .sort(comparator);
  476. const fakeMap = Object.create(null);
  477. for (const module of sortedModules) {
  478. const exportsType = module.getExportsType(
  479. moduleGraph,
  480. this.options.namespaceObject === "strict"
  481. );
  482. const id = chunkGraph.getModuleId(module);
  483. switch (exportsType) {
  484. case "namespace":
  485. fakeMap[id] = 9;
  486. hasType |= 1;
  487. break;
  488. case "dynamic":
  489. fakeMap[id] = 7;
  490. hasType |= 2;
  491. break;
  492. case "default-only":
  493. fakeMap[id] = 1;
  494. hasType |= 4;
  495. break;
  496. case "default-with-named":
  497. fakeMap[id] = 3;
  498. hasType |= 8;
  499. break;
  500. default:
  501. throw new Error(`Unexpected exports type ${exportsType}`);
  502. }
  503. }
  504. if (hasType === 1) {
  505. return 9;
  506. }
  507. if (hasType === 2) {
  508. return 7;
  509. }
  510. if (hasType === 4) {
  511. return 1;
  512. }
  513. if (hasType === 8) {
  514. return 3;
  515. }
  516. if (hasType === 0) {
  517. return 9;
  518. }
  519. return fakeMap;
  520. }
  521. getFakeMapInitStatement(fakeMap) {
  522. return typeof fakeMap === "object"
  523. ? `var fakeMap = ${JSON.stringify(fakeMap, null, "\t")};`
  524. : "";
  525. }
  526. getReturn(type, asyncModule) {
  527. if (type === 9) {
  528. return "__webpack_require__(id)";
  529. }
  530. return `${RuntimeGlobals.createFakeNamespaceObject}(id, ${type}${
  531. asyncModule ? " | 16" : ""
  532. })`;
  533. }
  534. getReturnModuleObjectSource(
  535. fakeMap,
  536. asyncModule,
  537. fakeMapDataExpression = "fakeMap[id]"
  538. ) {
  539. if (typeof fakeMap === "number") {
  540. return `return ${this.getReturn(fakeMap, asyncModule)};`;
  541. }
  542. return `return ${
  543. RuntimeGlobals.createFakeNamespaceObject
  544. }(id, ${fakeMapDataExpression}${asyncModule ? " | 16" : ""})`;
  545. }
  546. /**
  547. * @param {TODO} dependencies TODO
  548. * @param {TODO} id TODO
  549. * @param {ChunkGraph} chunkGraph the chunk graph
  550. * @returns {string} source code
  551. */
  552. getSyncSource(dependencies, id, chunkGraph) {
  553. const map = this.getUserRequestMap(dependencies, chunkGraph);
  554. const fakeMap = this.getFakeMap(dependencies, chunkGraph);
  555. const returnModuleObject = this.getReturnModuleObjectSource(fakeMap);
  556. return `var map = ${JSON.stringify(map, null, "\t")};
  557. ${this.getFakeMapInitStatement(fakeMap)}
  558. function webpackContext(req) {
  559. var id = webpackContextResolve(req);
  560. ${returnModuleObject}
  561. }
  562. function webpackContextResolve(req) {
  563. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  564. var e = new Error("Cannot find module '" + req + "'");
  565. e.code = 'MODULE_NOT_FOUND';
  566. throw e;
  567. }
  568. return map[req];
  569. }
  570. webpackContext.keys = function webpackContextKeys() {
  571. return Object.keys(map);
  572. };
  573. webpackContext.resolve = webpackContextResolve;
  574. module.exports = webpackContext;
  575. webpackContext.id = ${JSON.stringify(id)};`;
  576. }
  577. /**
  578. * @param {TODO} dependencies TODO
  579. * @param {TODO} id TODO
  580. * @param {ChunkGraph} chunkGraph the chunk graph
  581. * @returns {string} source code
  582. */
  583. getWeakSyncSource(dependencies, id, chunkGraph) {
  584. const map = this.getUserRequestMap(dependencies, chunkGraph);
  585. const fakeMap = this.getFakeMap(dependencies, chunkGraph);
  586. const returnModuleObject = this.getReturnModuleObjectSource(fakeMap);
  587. return `var map = ${JSON.stringify(map, null, "\t")};
  588. ${this.getFakeMapInitStatement(fakeMap)}
  589. function webpackContext(req) {
  590. var id = webpackContextResolve(req);
  591. if(!${RuntimeGlobals.moduleFactories}[id]) {
  592. var e = new Error("Module '" + req + "' ('" + id + "') is not available (weak dependency)");
  593. e.code = 'MODULE_NOT_FOUND';
  594. throw e;
  595. }
  596. ${returnModuleObject}
  597. }
  598. function webpackContextResolve(req) {
  599. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  600. var e = new Error("Cannot find module '" + req + "'");
  601. e.code = 'MODULE_NOT_FOUND';
  602. throw e;
  603. }
  604. return map[req];
  605. }
  606. webpackContext.keys = function webpackContextKeys() {
  607. return Object.keys(map);
  608. };
  609. webpackContext.resolve = webpackContextResolve;
  610. webpackContext.id = ${JSON.stringify(id)};
  611. module.exports = webpackContext;`;
  612. }
  613. /**
  614. * @param {TODO} dependencies TODO
  615. * @param {TODO} id TODO
  616. * @param {Object} context context
  617. * @param {ChunkGraph} context.chunkGraph the chunk graph
  618. * @param {RuntimeTemplate} context.runtimeTemplate the chunk graph
  619. * @returns {string} source code
  620. */
  621. getAsyncWeakSource(dependencies, id, { chunkGraph, runtimeTemplate }) {
  622. const arrow = runtimeTemplate.supportsArrowFunction();
  623. const map = this.getUserRequestMap(dependencies, chunkGraph);
  624. const fakeMap = this.getFakeMap(dependencies, chunkGraph);
  625. const returnModuleObject = this.getReturnModuleObjectSource(fakeMap, true);
  626. return `var map = ${JSON.stringify(map, null, "\t")};
  627. ${this.getFakeMapInitStatement(fakeMap)}
  628. function webpackAsyncContext(req) {
  629. return webpackAsyncContextResolve(req).then(${
  630. arrow ? "id =>" : "function(id)"
  631. } {
  632. if(!${RuntimeGlobals.moduleFactories}[id]) {
  633. var e = new Error("Module '" + req + "' ('" + id + "') is not available (weak dependency)");
  634. e.code = 'MODULE_NOT_FOUND';
  635. throw e;
  636. }
  637. ${returnModuleObject}
  638. });
  639. }
  640. function webpackAsyncContextResolve(req) {
  641. // Here Promise.resolve().then() is used instead of new Promise() to prevent
  642. // uncaught exception popping up in devtools
  643. return Promise.resolve().then(${arrow ? "() =>" : "function()"} {
  644. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  645. var e = new Error("Cannot find module '" + req + "'");
  646. e.code = 'MODULE_NOT_FOUND';
  647. throw e;
  648. }
  649. return map[req];
  650. });
  651. }
  652. webpackAsyncContext.keys = ${runtimeTemplate.returningFunction(
  653. "Object.keys(map)"
  654. )};
  655. webpackAsyncContext.resolve = webpackAsyncContextResolve;
  656. webpackAsyncContext.id = ${JSON.stringify(id)};
  657. module.exports = webpackAsyncContext;`;
  658. }
  659. /**
  660. * @param {TODO} dependencies TODO
  661. * @param {TODO} id TODO
  662. * @param {Object} context context
  663. * @param {ChunkGraph} context.chunkGraph the chunk graph
  664. * @param {RuntimeTemplate} context.runtimeTemplate the chunk graph
  665. * @returns {string} source code
  666. */
  667. getEagerSource(dependencies, id, { chunkGraph, runtimeTemplate }) {
  668. const arrow = runtimeTemplate.supportsArrowFunction();
  669. const map = this.getUserRequestMap(dependencies, chunkGraph);
  670. const fakeMap = this.getFakeMap(dependencies, chunkGraph);
  671. const thenFunction =
  672. fakeMap !== 9
  673. ? `${arrow ? "id =>" : "function(id)"} {
  674. ${this.getReturnModuleObjectSource(fakeMap)}
  675. }`
  676. : "__webpack_require__";
  677. return `var map = ${JSON.stringify(map, null, "\t")};
  678. ${this.getFakeMapInitStatement(fakeMap)}
  679. function webpackAsyncContext(req) {
  680. return webpackAsyncContextResolve(req).then(${thenFunction});
  681. }
  682. function webpackAsyncContextResolve(req) {
  683. // Here Promise.resolve().then() is used instead of new Promise() to prevent
  684. // uncaught exception popping up in devtools
  685. return Promise.resolve().then(${arrow ? "() =>" : "function()"} {
  686. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  687. var e = new Error("Cannot find module '" + req + "'");
  688. e.code = 'MODULE_NOT_FOUND';
  689. throw e;
  690. }
  691. return map[req];
  692. });
  693. }
  694. webpackAsyncContext.keys = ${runtimeTemplate.returningFunction(
  695. "Object.keys(map)"
  696. )};
  697. webpackAsyncContext.resolve = webpackAsyncContextResolve;
  698. webpackAsyncContext.id = ${JSON.stringify(id)};
  699. module.exports = webpackAsyncContext;`;
  700. }
  701. /**
  702. * @param {TODO} block TODO
  703. * @param {TODO} dependencies TODO
  704. * @param {TODO} id TODO
  705. * @param {Object} options options object
  706. * @param {RuntimeTemplate} options.runtimeTemplate the runtime template
  707. * @param {ChunkGraph} options.chunkGraph the chunk graph
  708. * @returns {string} source code
  709. */
  710. getLazyOnceSource(block, dependencies, id, { runtimeTemplate, chunkGraph }) {
  711. const promise = runtimeTemplate.blockPromise({
  712. chunkGraph,
  713. block,
  714. message: "lazy-once context",
  715. runtimeRequirements: new Set()
  716. });
  717. const arrow = runtimeTemplate.supportsArrowFunction();
  718. const map = this.getUserRequestMap(dependencies, chunkGraph);
  719. const fakeMap = this.getFakeMap(dependencies, chunkGraph);
  720. const thenFunction =
  721. fakeMap !== 9
  722. ? `${arrow ? "id =>" : "function(id)"} {
  723. ${this.getReturnModuleObjectSource(fakeMap, true)};
  724. }`
  725. : "__webpack_require__";
  726. return `var map = ${JSON.stringify(map, null, "\t")};
  727. ${this.getFakeMapInitStatement(fakeMap)}
  728. function webpackAsyncContext(req) {
  729. return webpackAsyncContextResolve(req).then(${thenFunction});
  730. }
  731. function webpackAsyncContextResolve(req) {
  732. return ${promise}.then(${arrow ? "() =>" : "function()"} {
  733. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  734. var e = new Error("Cannot find module '" + req + "'");
  735. e.code = 'MODULE_NOT_FOUND';
  736. throw e;
  737. }
  738. return map[req];
  739. });
  740. }
  741. webpackAsyncContext.keys = ${runtimeTemplate.returningFunction(
  742. "Object.keys(map)"
  743. )};
  744. webpackAsyncContext.resolve = webpackAsyncContextResolve;
  745. webpackAsyncContext.id = ${JSON.stringify(id)};
  746. module.exports = webpackAsyncContext;`;
  747. }
  748. /**
  749. * @param {TODO} blocks TODO
  750. * @param {TODO} id TODO
  751. * @param {Object} context context
  752. * @param {ChunkGraph} context.chunkGraph the chunk graph
  753. * @param {RuntimeTemplate} context.runtimeTemplate the chunk graph
  754. * @returns {string} source code
  755. */
  756. getLazySource(blocks, id, { chunkGraph, runtimeTemplate }) {
  757. const moduleGraph = chunkGraph.moduleGraph;
  758. const arrow = runtimeTemplate.supportsArrowFunction();
  759. let hasMultipleOrNoChunks = false;
  760. let hasNoChunk = true;
  761. const fakeMap = this.getFakeMap(
  762. blocks.map(b => b.dependencies[0]),
  763. chunkGraph
  764. );
  765. const hasFakeMap = typeof fakeMap === "object";
  766. const items = blocks
  767. .map(block => {
  768. const dependency = block.dependencies[0];
  769. return {
  770. dependency: dependency,
  771. module: moduleGraph.getModule(dependency),
  772. block: block,
  773. userRequest: dependency.userRequest,
  774. chunks: undefined
  775. };
  776. })
  777. .filter(item => item.module);
  778. for (const item of items) {
  779. const chunkGroup = chunkGraph.getBlockChunkGroup(item.block);
  780. const chunks = (chunkGroup && chunkGroup.chunks) || [];
  781. item.chunks = chunks;
  782. if (chunks.length > 0) {
  783. hasNoChunk = false;
  784. }
  785. if (chunks.length !== 1) {
  786. hasMultipleOrNoChunks = true;
  787. }
  788. }
  789. const shortMode = hasNoChunk && !hasFakeMap;
  790. const sortedItems = items.sort((a, b) => {
  791. if (a.userRequest === b.userRequest) return 0;
  792. return a.userRequest < b.userRequest ? -1 : 1;
  793. });
  794. const map = Object.create(null);
  795. for (const item of sortedItems) {
  796. const moduleId = chunkGraph.getModuleId(item.module);
  797. if (shortMode) {
  798. map[item.userRequest] = moduleId;
  799. } else {
  800. const arrayStart = [moduleId];
  801. if (hasFakeMap) {
  802. arrayStart.push(fakeMap[moduleId]);
  803. }
  804. map[item.userRequest] = arrayStart.concat(
  805. item.chunks.map(chunk => chunk.id)
  806. );
  807. }
  808. }
  809. const chunksStartPosition = hasFakeMap ? 2 : 1;
  810. const requestPrefix = hasNoChunk
  811. ? "Promise.resolve()"
  812. : hasMultipleOrNoChunks
  813. ? `Promise.all(ids.slice(${chunksStartPosition}).map(${RuntimeGlobals.ensureChunk}))`
  814. : `${RuntimeGlobals.ensureChunk}(ids[${chunksStartPosition}])`;
  815. const returnModuleObject = this.getReturnModuleObjectSource(
  816. fakeMap,
  817. true,
  818. shortMode ? "invalid" : "ids[1]"
  819. );
  820. const webpackAsyncContext =
  821. requestPrefix === "Promise.resolve()"
  822. ? `
  823. function webpackAsyncContext(req) {
  824. return Promise.resolve().then(${arrow ? "() =>" : "function()"} {
  825. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  826. var e = new Error("Cannot find module '" + req + "'");
  827. e.code = 'MODULE_NOT_FOUND';
  828. throw e;
  829. }
  830. ${shortMode ? "var id = map[req];" : "var ids = map[req], id = ids[0];"}
  831. ${returnModuleObject}
  832. });
  833. }`
  834. : `function webpackAsyncContext(req) {
  835. if(!${RuntimeGlobals.hasOwnProperty}(map, req)) {
  836. return Promise.resolve().then(${arrow ? "() =>" : "function()"} {
  837. var e = new Error("Cannot find module '" + req + "'");
  838. e.code = 'MODULE_NOT_FOUND';
  839. throw e;
  840. });
  841. }
  842. var ids = map[req], id = ids[0];
  843. return ${requestPrefix}.then(${arrow ? "() =>" : "function()"} {
  844. ${returnModuleObject}
  845. });
  846. }`;
  847. return `var map = ${JSON.stringify(map, null, "\t")};
  848. ${webpackAsyncContext}
  849. webpackAsyncContext.keys = ${runtimeTemplate.returningFunction(
  850. "Object.keys(map)"
  851. )};
  852. webpackAsyncContext.id = ${JSON.stringify(id)};
  853. module.exports = webpackAsyncContext;`;
  854. }
  855. getSourceForEmptyContext(id, runtimeTemplate) {
  856. return `function webpackEmptyContext(req) {
  857. var e = new Error("Cannot find module '" + req + "'");
  858. e.code = 'MODULE_NOT_FOUND';
  859. throw e;
  860. }
  861. webpackEmptyContext.keys = ${runtimeTemplate.returningFunction("[]")};
  862. webpackEmptyContext.resolve = webpackEmptyContext;
  863. webpackEmptyContext.id = ${JSON.stringify(id)};
  864. module.exports = webpackEmptyContext;`;
  865. }
  866. getSourceForEmptyAsyncContext(id, runtimeTemplate) {
  867. const arrow = runtimeTemplate.supportsArrowFunction();
  868. return `function webpackEmptyAsyncContext(req) {
  869. // Here Promise.resolve().then() is used instead of new Promise() to prevent
  870. // uncaught exception popping up in devtools
  871. return Promise.resolve().then(${arrow ? "() =>" : "function()"} {
  872. var e = new Error("Cannot find module '" + req + "'");
  873. e.code = 'MODULE_NOT_FOUND';
  874. throw e;
  875. });
  876. }
  877. webpackEmptyAsyncContext.keys = ${runtimeTemplate.returningFunction("[]")};
  878. webpackEmptyAsyncContext.resolve = webpackEmptyAsyncContext;
  879. webpackEmptyAsyncContext.id = ${JSON.stringify(id)};
  880. module.exports = webpackEmptyAsyncContext;`;
  881. }
  882. /**
  883. * @param {string} asyncMode module mode
  884. * @param {CodeGenerationContext} context context info
  885. * @returns {string} the source code
  886. */
  887. getSourceString(asyncMode, { runtimeTemplate, chunkGraph }) {
  888. const id = chunkGraph.getModuleId(this);
  889. if (asyncMode === "lazy") {
  890. if (this.blocks && this.blocks.length > 0) {
  891. return this.getLazySource(this.blocks, id, {
  892. runtimeTemplate,
  893. chunkGraph
  894. });
  895. }
  896. return this.getSourceForEmptyAsyncContext(id, runtimeTemplate);
  897. }
  898. if (asyncMode === "eager") {
  899. if (this.dependencies && this.dependencies.length > 0) {
  900. return this.getEagerSource(this.dependencies, id, {
  901. chunkGraph,
  902. runtimeTemplate
  903. });
  904. }
  905. return this.getSourceForEmptyAsyncContext(id, runtimeTemplate);
  906. }
  907. if (asyncMode === "lazy-once") {
  908. const block = this.blocks[0];
  909. if (block) {
  910. return this.getLazyOnceSource(block, block.dependencies, id, {
  911. runtimeTemplate,
  912. chunkGraph
  913. });
  914. }
  915. return this.getSourceForEmptyAsyncContext(id, runtimeTemplate);
  916. }
  917. if (asyncMode === "async-weak") {
  918. if (this.dependencies && this.dependencies.length > 0) {
  919. return this.getAsyncWeakSource(this.dependencies, id, {
  920. chunkGraph,
  921. runtimeTemplate
  922. });
  923. }
  924. return this.getSourceForEmptyAsyncContext(id, runtimeTemplate);
  925. }
  926. if (asyncMode === "weak") {
  927. if (this.dependencies && this.dependencies.length > 0) {
  928. return this.getWeakSyncSource(this.dependencies, id, chunkGraph);
  929. }
  930. }
  931. if (this.dependencies && this.dependencies.length > 0) {
  932. return this.getSyncSource(this.dependencies, id, chunkGraph);
  933. }
  934. return this.getSourceForEmptyContext(id, runtimeTemplate);
  935. }
  936. getSource(sourceString) {
  937. if (this.useSourceMap || this.useSimpleSourceMap) {
  938. return new OriginalSource(sourceString, this.identifier());
  939. }
  940. return new RawSource(sourceString);
  941. }
  942. /**
  943. * @param {CodeGenerationContext} context context for code generation
  944. * @returns {CodeGenerationResult} result
  945. */
  946. codeGeneration(context) {
  947. const { chunkGraph } = context;
  948. const sources = new Map();
  949. sources.set(
  950. "javascript",
  951. this.getSource(this.getSourceString(this.options.mode, context))
  952. );
  953. const set = new Set();
  954. const allDeps = /** @type {ContextElementDependency[]} */ (
  955. this.dependencies.concat(this.blocks.map(b => b.dependencies[0]))
  956. );
  957. set.add(RuntimeGlobals.module);
  958. set.add(RuntimeGlobals.hasOwnProperty);
  959. if (allDeps.length > 0) {
  960. const asyncMode = this.options.mode;
  961. set.add(RuntimeGlobals.require);
  962. if (asyncMode === "weak") {
  963. set.add(RuntimeGlobals.moduleFactories);
  964. } else if (asyncMode === "async-weak") {
  965. set.add(RuntimeGlobals.moduleFactories);
  966. set.add(RuntimeGlobals.ensureChunk);
  967. } else if (asyncMode === "lazy" || asyncMode === "lazy-once") {
  968. set.add(RuntimeGlobals.ensureChunk);
  969. }
  970. if (this.getFakeMap(allDeps, chunkGraph) !== 9) {
  971. set.add(RuntimeGlobals.createFakeNamespaceObject);
  972. }
  973. }
  974. return {
  975. sources,
  976. runtimeRequirements: set
  977. };
  978. }
  979. /**
  980. * @param {string=} type the source type for which the size should be estimated
  981. * @returns {number} the estimated size of the module (must be non-zero)
  982. */
  983. size(type) {
  984. // base penalty
  985. let size = 160;
  986. // if we don't have dependencies we stop here.
  987. for (const dependency of this.dependencies) {
  988. const element = /** @type {ContextElementDependency} */ (dependency);
  989. size += 5 + element.userRequest.length;
  990. }
  991. return size;
  992. }
  993. serialize(context) {
  994. const { write } = context;
  995. write(this._identifier);
  996. write(this._forceBuild);
  997. super.serialize(context);
  998. }
  999. deserialize(context) {
  1000. const { read } = context;
  1001. this._identifier = read();
  1002. this._forceBuild = read();
  1003. super.deserialize(context);
  1004. }
  1005. }
  1006. makeSerializable(ContextModule, "webpack/lib/ContextModule");
  1007. module.exports = ContextModule;