ImportMetaContextDependencyParserPlugin.js 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Ivan Kopeykin @vankop
  4. */
  5. "use strict";
  6. const WebpackError = require("../WebpackError");
  7. const {
  8. evaluateToIdentifier
  9. } = require("../javascript/JavascriptParserHelpers");
  10. const ImportMetaContextDependency = require("./ImportMetaContextDependency");
  11. /** @typedef {import("estree").Expression} Expression */
  12. /** @typedef {import("estree").ObjectExpression} ObjectExpression */
  13. /** @typedef {import("estree").Property} Property */
  14. /** @typedef {import("estree").Identifier} Identifier */
  15. /** @typedef {import("../javascript/JavascriptParser")} JavascriptParser */
  16. /** @typedef {import("../javascript/JavascriptParser").Range} Range */
  17. /** @typedef {import("../ContextModule").ContextModuleOptions} ContextModuleOptions */
  18. /** @typedef {import("../ContextModule").ContextMode} ContextMode */
  19. /** @typedef {import("../Chunk").ChunkName} ChunkName */
  20. /** @typedef {import("../ChunkGroup").RawChunkGroupOptions} RawChunkGroupOptions */
  21. /** @typedef {import("../Dependency").RawReferencedExports} RawReferencedExports */
  22. /** @typedef {import("../Dependency").DependencyLocation} DependencyLocation */
  23. /** @typedef {import("../javascript/BasicEvaluatedExpression")} BasicEvaluatedExpression */
  24. /** @typedef {Pick<ContextModuleOptions, "mode" | "recursive" | "regExp" | "include" | "exclude" | "chunkName"> & { groupOptions: RawChunkGroupOptions, exports?: RawReferencedExports }} ImportMetaContextOptions */
  25. /**
  26. * Creates a property parse error.
  27. * @param {Property} prop property
  28. * @param {string} expect except message
  29. * @returns {WebpackError} error
  30. */
  31. function createPropertyParseError(prop, expect) {
  32. return createError(
  33. `Parsing import.meta.webpackContext options failed. Unknown value for property ${JSON.stringify(
  34. /** @type {Identifier} */
  35. (prop.key).name
  36. )}, expected type ${expect}.`,
  37. /** @type {DependencyLocation} */
  38. (prop.value.loc)
  39. );
  40. }
  41. /**
  42. * Creates an error from the provided msg.
  43. * @param {string} msg message
  44. * @param {DependencyLocation} loc location
  45. * @returns {WebpackError} error
  46. */
  47. function createError(msg, loc) {
  48. const error = new WebpackError(msg);
  49. error.name = "ImportMetaContextError";
  50. error.loc = loc;
  51. return error;
  52. }
  53. const PLUGIN_NAME = "ImportMetaContextDependencyParserPlugin";
  54. module.exports = class ImportMetaContextDependencyParserPlugin {
  55. /**
  56. * Applies the plugin by registering its hooks on the compiler.
  57. * @param {JavascriptParser} parser the parser
  58. * @returns {void}
  59. */
  60. apply(parser) {
  61. parser.hooks.evaluateIdentifier
  62. .for("import.meta.webpackContext")
  63. .tap(PLUGIN_NAME, (expr) =>
  64. evaluateToIdentifier(
  65. "import.meta.webpackContext",
  66. "import.meta",
  67. () => ["webpackContext"],
  68. true
  69. )(expr)
  70. );
  71. parser.hooks.call
  72. .for("import.meta.webpackContext")
  73. .tap(PLUGIN_NAME, (expr) => {
  74. if (expr.arguments.length < 1 || expr.arguments.length > 2) return;
  75. const [directoryNode, optionsNode] = expr.arguments;
  76. if (optionsNode && optionsNode.type !== "ObjectExpression") return;
  77. const requestExpr = parser.evaluateExpression(
  78. /** @type {Expression} */ (directoryNode)
  79. );
  80. if (!requestExpr.isString()) return;
  81. const request = /** @type {string} */ (requestExpr.string);
  82. /** @type {WebpackError[]} */
  83. const errors = [];
  84. let regExp = /^\.\/.*$/;
  85. let recursive = true;
  86. /** @type {ContextMode} */
  87. let mode = "sync";
  88. /** @type {ContextModuleOptions["include"]} */
  89. let include;
  90. /** @type {ContextModuleOptions["exclude"]} */
  91. let exclude;
  92. /** @type {RawChunkGroupOptions} */
  93. const groupOptions = {};
  94. /** @type {ChunkName | undefined} */
  95. let chunkName;
  96. /** @type {RawReferencedExports | undefined} */
  97. let exports;
  98. if (optionsNode) {
  99. for (const prop of /** @type {ObjectExpression} */ (optionsNode)
  100. .properties) {
  101. if (prop.type !== "Property" || prop.key.type !== "Identifier") {
  102. errors.push(
  103. createError(
  104. "Parsing import.meta.webpackContext options failed.",
  105. /** @type {DependencyLocation} */
  106. (optionsNode.loc)
  107. )
  108. );
  109. break;
  110. }
  111. switch (prop.key.name) {
  112. case "regExp": {
  113. const regExpExpr = parser.evaluateExpression(
  114. /** @type {Expression} */ (prop.value)
  115. );
  116. if (!regExpExpr.isRegExp()) {
  117. errors.push(createPropertyParseError(prop, "RegExp"));
  118. } else {
  119. regExp = /** @type {RegExp} */ (regExpExpr.regExp);
  120. }
  121. break;
  122. }
  123. case "include": {
  124. const regExpExpr = parser.evaluateExpression(
  125. /** @type {Expression} */ (prop.value)
  126. );
  127. if (!regExpExpr.isRegExp()) {
  128. errors.push(createPropertyParseError(prop, "RegExp"));
  129. } else {
  130. include = regExpExpr.regExp;
  131. }
  132. break;
  133. }
  134. case "exclude": {
  135. const regExpExpr = parser.evaluateExpression(
  136. /** @type {Expression} */ (prop.value)
  137. );
  138. if (!regExpExpr.isRegExp()) {
  139. errors.push(createPropertyParseError(prop, "RegExp"));
  140. } else {
  141. exclude = regExpExpr.regExp;
  142. }
  143. break;
  144. }
  145. case "mode": {
  146. const modeExpr = parser.evaluateExpression(
  147. /** @type {Expression} */ (prop.value)
  148. );
  149. if (!modeExpr.isString()) {
  150. errors.push(createPropertyParseError(prop, "string"));
  151. } else {
  152. mode = /** @type {ContextModuleOptions["mode"]} */ (
  153. modeExpr.string
  154. );
  155. }
  156. break;
  157. }
  158. case "chunkName": {
  159. const expr = parser.evaluateExpression(
  160. /** @type {Expression} */ (prop.value)
  161. );
  162. if (!expr.isString()) {
  163. errors.push(createPropertyParseError(prop, "string"));
  164. } else {
  165. chunkName = expr.string;
  166. }
  167. break;
  168. }
  169. case "exports": {
  170. const expr = parser.evaluateExpression(
  171. /** @type {Expression} */ (prop.value)
  172. );
  173. if (expr.isString()) {
  174. exports = [[/** @type {string} */ (expr.string)]];
  175. } else if (expr.isArray()) {
  176. const items =
  177. /** @type {BasicEvaluatedExpression[]} */
  178. (expr.items);
  179. if (
  180. items.every((i) => {
  181. if (!i.isArray()) return false;
  182. const innerItems =
  183. /** @type {BasicEvaluatedExpression[]} */ (i.items);
  184. return innerItems.every((i) => i.isString());
  185. })
  186. ) {
  187. exports = [];
  188. for (const i1 of items) {
  189. /** @type {string[]} */
  190. const export_ = [];
  191. for (const i2 of /** @type {BasicEvaluatedExpression[]} */ (
  192. i1.items
  193. )) {
  194. export_.push(/** @type {string} */ (i2.string));
  195. }
  196. exports.push(export_);
  197. }
  198. } else {
  199. errors.push(
  200. createPropertyParseError(prop, "string|string[][]")
  201. );
  202. }
  203. } else {
  204. errors.push(
  205. createPropertyParseError(prop, "string|string[][]")
  206. );
  207. }
  208. break;
  209. }
  210. case "prefetch": {
  211. const expr = parser.evaluateExpression(
  212. /** @type {Expression} */ (prop.value)
  213. );
  214. if (expr.isBoolean()) {
  215. groupOptions.prefetchOrder = 0;
  216. } else if (expr.isNumber()) {
  217. groupOptions.prefetchOrder = expr.number;
  218. } else {
  219. errors.push(createPropertyParseError(prop, "boolean|number"));
  220. }
  221. break;
  222. }
  223. case "preload": {
  224. const expr = parser.evaluateExpression(
  225. /** @type {Expression} */ (prop.value)
  226. );
  227. if (expr.isBoolean()) {
  228. groupOptions.preloadOrder = 0;
  229. } else if (expr.isNumber()) {
  230. groupOptions.preloadOrder = expr.number;
  231. } else {
  232. errors.push(createPropertyParseError(prop, "boolean|number"));
  233. }
  234. break;
  235. }
  236. case "fetchPriority": {
  237. const expr = parser.evaluateExpression(
  238. /** @type {Expression} */ (prop.value)
  239. );
  240. if (
  241. expr.isString() &&
  242. ["high", "low", "auto"].includes(
  243. /** @type {string} */ (expr.string)
  244. )
  245. ) {
  246. groupOptions.fetchPriority =
  247. /** @type {RawChunkGroupOptions["fetchPriority"]} */ (
  248. expr.string
  249. );
  250. } else {
  251. errors.push(
  252. createPropertyParseError(prop, '"high"|"low"|"auto"')
  253. );
  254. }
  255. break;
  256. }
  257. case "recursive": {
  258. const recursiveExpr = parser.evaluateExpression(
  259. /** @type {Expression} */ (prop.value)
  260. );
  261. if (!recursiveExpr.isBoolean()) {
  262. errors.push(createPropertyParseError(prop, "boolean"));
  263. } else {
  264. recursive = /** @type {boolean} */ (recursiveExpr.bool);
  265. }
  266. break;
  267. }
  268. default:
  269. errors.push(
  270. createError(
  271. `Parsing import.meta.webpackContext options failed. Unknown property ${JSON.stringify(
  272. prop.key.name
  273. )}.`,
  274. /** @type {DependencyLocation} */ (optionsNode.loc)
  275. )
  276. );
  277. }
  278. }
  279. }
  280. if (errors.length) {
  281. for (const error of errors) parser.state.current.addError(error);
  282. return;
  283. }
  284. const dep = new ImportMetaContextDependency(
  285. {
  286. request,
  287. include,
  288. exclude,
  289. recursive,
  290. regExp,
  291. groupOptions,
  292. chunkName,
  293. referencedExports: exports,
  294. mode,
  295. category: "esm"
  296. },
  297. /** @type {Range} */ (expr.range)
  298. );
  299. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  300. dep.optional = Boolean(parser.scope.inTry);
  301. parser.state.current.addDependency(dep);
  302. return true;
  303. });
  304. }
  305. };