10
0

AMDRequireDependenciesBlockParserPlugin.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const RuntimeGlobals = require("../RuntimeGlobals");
  7. const UnsupportedFeatureWarning = require("../UnsupportedFeatureWarning");
  8. const AMDRequireArrayDependency = require("./AMDRequireArrayDependency");
  9. const AMDRequireContextDependency = require("./AMDRequireContextDependency");
  10. const AMDRequireDependenciesBlock = require("./AMDRequireDependenciesBlock");
  11. const AMDRequireDependency = require("./AMDRequireDependency");
  12. const AMDRequireItemDependency = require("./AMDRequireItemDependency");
  13. const ConstDependency = require("./ConstDependency");
  14. const ContextDependencyHelpers = require("./ContextDependencyHelpers");
  15. const LocalModuleDependency = require("./LocalModuleDependency");
  16. const { getLocalModule } = require("./LocalModulesHelpers");
  17. const UnsupportedDependency = require("./UnsupportedDependency");
  18. const getFunctionExpression = require("./getFunctionExpression");
  19. /** @typedef {import("estree").CallExpression} CallExpression */
  20. /** @typedef {import("estree").Expression} Expression */
  21. /** @typedef {import("estree").Identifier} Identifier */
  22. /** @typedef {import("estree").SourceLocation} SourceLocation */
  23. /** @typedef {import("estree").SpreadElement} SpreadElement */
  24. /** @typedef {import("../../declarations/WebpackOptions").JavascriptParserOptions} JavascriptParserOptions */
  25. /** @typedef {import("../Dependency").DependencyLocation} DependencyLocation */
  26. /** @typedef {import("../Module").BuildInfo} BuildInfo */
  27. /** @typedef {import("../javascript/BasicEvaluatedExpression")} BasicEvaluatedExpression */
  28. /** @typedef {import("../javascript/JavascriptParser")} JavascriptParser */
  29. /** @typedef {import("../javascript/JavascriptParser").Range} Range */
  30. class AMDRequireDependenciesBlockParserPlugin {
  31. /**
  32. * @param {JavascriptParserOptions} options parserOptions
  33. */
  34. constructor(options) {
  35. this.options = options;
  36. }
  37. /**
  38. * @param {JavascriptParser} parser the parser
  39. * @param {Expression | SpreadElement} expression expression
  40. * @returns {boolean} need bind this
  41. */
  42. processFunctionArgument(parser, expression) {
  43. let bindThis = true;
  44. const fnData = getFunctionExpression(expression);
  45. if (fnData) {
  46. parser.inScope(
  47. fnData.fn.params.filter(
  48. i =>
  49. !["require", "module", "exports"].includes(
  50. /** @type {Identifier} */ (i).name
  51. )
  52. ),
  53. () => {
  54. if (fnData.fn.body.type === "BlockStatement") {
  55. parser.walkStatement(fnData.fn.body);
  56. } else {
  57. parser.walkExpression(fnData.fn.body);
  58. }
  59. }
  60. );
  61. parser.walkExpressions(fnData.expressions);
  62. if (fnData.needThis === false) {
  63. bindThis = false;
  64. }
  65. } else {
  66. parser.walkExpression(expression);
  67. }
  68. return bindThis;
  69. }
  70. /**
  71. * @param {JavascriptParser} parser the parser
  72. * @returns {void}
  73. */
  74. apply(parser) {
  75. parser.hooks.call
  76. .for("require")
  77. .tap(
  78. "AMDRequireDependenciesBlockParserPlugin",
  79. this.processCallRequire.bind(this, parser)
  80. );
  81. }
  82. /**
  83. * @param {JavascriptParser} parser the parser
  84. * @param {CallExpression} expr call expression
  85. * @param {BasicEvaluatedExpression} param param
  86. * @returns {boolean | undefined} result
  87. */
  88. processArray(parser, expr, param) {
  89. if (param.isArray()) {
  90. for (const p of /** @type {BasicEvaluatedExpression[]} */ (param.items)) {
  91. const result = this.processItem(parser, expr, p);
  92. if (result === undefined) {
  93. this.processContext(parser, expr, p);
  94. }
  95. }
  96. return true;
  97. } else if (param.isConstArray()) {
  98. /** @type {(string | LocalModuleDependency | AMDRequireItemDependency)[]} */
  99. const deps = [];
  100. for (const request of /** @type {any[]} */ (param.array)) {
  101. let dep;
  102. let localModule;
  103. if (request === "require") {
  104. dep = RuntimeGlobals.require;
  105. } else if (["exports", "module"].includes(request)) {
  106. dep = request;
  107. } else if ((localModule = getLocalModule(parser.state, request))) {
  108. localModule.flagUsed();
  109. dep = new LocalModuleDependency(localModule, undefined, false);
  110. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  111. parser.state.module.addPresentationalDependency(dep);
  112. } else {
  113. dep = this.newRequireItemDependency(request);
  114. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  115. dep.optional = Boolean(parser.scope.inTry);
  116. parser.state.current.addDependency(dep);
  117. }
  118. deps.push(dep);
  119. }
  120. const dep = this.newRequireArrayDependency(
  121. deps,
  122. /** @type {Range} */ (param.range)
  123. );
  124. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  125. dep.optional = Boolean(parser.scope.inTry);
  126. parser.state.module.addPresentationalDependency(dep);
  127. return true;
  128. }
  129. }
  130. /**
  131. * @param {JavascriptParser} parser the parser
  132. * @param {CallExpression} expr call expression
  133. * @param {BasicEvaluatedExpression} param param
  134. * @returns {boolean | undefined} result
  135. */
  136. processItem(parser, expr, param) {
  137. if (param.isConditional()) {
  138. for (const p of /** @type {BasicEvaluatedExpression[]} */ (
  139. param.options
  140. )) {
  141. const result = this.processItem(parser, expr, p);
  142. if (result === undefined) {
  143. this.processContext(parser, expr, p);
  144. }
  145. }
  146. return true;
  147. } else if (param.isString()) {
  148. let dep;
  149. let localModule;
  150. if (param.string === "require") {
  151. dep = new ConstDependency(
  152. RuntimeGlobals.require,
  153. /** @type {TODO} */ (param.string),
  154. [RuntimeGlobals.require]
  155. );
  156. } else if (param.string === "module") {
  157. dep = new ConstDependency(
  158. /** @type {BuildInfo} */
  159. (parser.state.module.buildInfo).moduleArgument,
  160. /** @type {Range} */ (param.range),
  161. [RuntimeGlobals.module]
  162. );
  163. } else if (param.string === "exports") {
  164. dep = new ConstDependency(
  165. /** @type {BuildInfo} */
  166. (parser.state.module.buildInfo).exportsArgument,
  167. /** @type {Range} */ (param.range),
  168. [RuntimeGlobals.exports]
  169. );
  170. } else if (
  171. (localModule = getLocalModule(
  172. parser.state,
  173. /** @type {string} */ (param.string)
  174. ))
  175. ) {
  176. localModule.flagUsed();
  177. dep = new LocalModuleDependency(localModule, param.range, false);
  178. } else {
  179. dep = this.newRequireItemDependency(
  180. /** @type {string} */ (param.string),
  181. param.range
  182. );
  183. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  184. dep.optional = Boolean(parser.scope.inTry);
  185. parser.state.current.addDependency(dep);
  186. return true;
  187. }
  188. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  189. parser.state.module.addPresentationalDependency(dep);
  190. return true;
  191. }
  192. }
  193. /**
  194. * @param {JavascriptParser} parser the parser
  195. * @param {CallExpression} expr call expression
  196. * @param {BasicEvaluatedExpression} param param
  197. * @returns {boolean | undefined} result
  198. */
  199. processContext(parser, expr, param) {
  200. const dep = ContextDependencyHelpers.create(
  201. AMDRequireContextDependency,
  202. /** @type {Range} */ (param.range),
  203. param,
  204. expr,
  205. this.options,
  206. {
  207. category: "amd"
  208. },
  209. parser
  210. );
  211. if (!dep) return;
  212. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  213. dep.optional = Boolean(parser.scope.inTry);
  214. parser.state.current.addDependency(dep);
  215. return true;
  216. }
  217. /**
  218. * @param {BasicEvaluatedExpression} param param
  219. * @returns {string | undefined} result
  220. */
  221. processArrayForRequestString(param) {
  222. if (param.isArray()) {
  223. const result =
  224. /** @type {BasicEvaluatedExpression[]} */
  225. (param.items).map(item => this.processItemForRequestString(item));
  226. if (result.every(Boolean)) return result.join(" ");
  227. } else if (param.isConstArray()) {
  228. return /** @type {string[]} */ (param.array).join(" ");
  229. }
  230. }
  231. /**
  232. * @param {BasicEvaluatedExpression} param param
  233. * @returns {string | undefined} result
  234. */
  235. processItemForRequestString(param) {
  236. if (param.isConditional()) {
  237. const result =
  238. /** @type {BasicEvaluatedExpression[]} */
  239. (param.options).map(item => this.processItemForRequestString(item));
  240. if (result.every(Boolean)) return result.join("|");
  241. } else if (param.isString()) {
  242. return param.string;
  243. }
  244. }
  245. /**
  246. * @param {JavascriptParser} parser the parser
  247. * @param {CallExpression} expr call expression
  248. * @returns {boolean | undefined} result
  249. */
  250. processCallRequire(parser, expr) {
  251. /** @type {BasicEvaluatedExpression | undefined} */
  252. let param;
  253. /** @type {AMDRequireDependenciesBlock | undefined | null} */
  254. let depBlock;
  255. /** @type {AMDRequireDependency | undefined} */
  256. let dep;
  257. /** @type {boolean | undefined} */
  258. let result;
  259. const old = parser.state.current;
  260. if (expr.arguments.length >= 1) {
  261. param = parser.evaluateExpression(
  262. /** @type {Expression} */ (expr.arguments[0])
  263. );
  264. depBlock = this.newRequireDependenciesBlock(
  265. /** @type {DependencyLocation} */ (expr.loc),
  266. this.processArrayForRequestString(param)
  267. );
  268. dep = this.newRequireDependency(
  269. /** @type {Range} */ (expr.range),
  270. /** @type {Range} */ (param.range),
  271. expr.arguments.length > 1
  272. ? /** @type {Range} */ (expr.arguments[1].range)
  273. : null,
  274. expr.arguments.length > 2
  275. ? /** @type {Range} */ (expr.arguments[2].range)
  276. : null
  277. );
  278. dep.loc = /** @type {DependencyLocation} */ (expr.loc);
  279. depBlock.addDependency(dep);
  280. parser.state.current = /** @type {TODO} */ (depBlock);
  281. }
  282. if (expr.arguments.length === 1) {
  283. parser.inScope([], () => {
  284. result = this.processArray(
  285. parser,
  286. expr,
  287. /** @type {BasicEvaluatedExpression} */ (param)
  288. );
  289. });
  290. parser.state.current = old;
  291. if (!result) return;
  292. parser.state.current.addBlock(
  293. /** @type {AMDRequireDependenciesBlock} */ (depBlock)
  294. );
  295. return true;
  296. }
  297. if (expr.arguments.length === 2 || expr.arguments.length === 3) {
  298. try {
  299. parser.inScope([], () => {
  300. result = this.processArray(
  301. parser,
  302. expr,
  303. /** @type {BasicEvaluatedExpression} */ (param)
  304. );
  305. });
  306. if (!result) {
  307. const dep = new UnsupportedDependency(
  308. "unsupported",
  309. /** @type {Range} */ (expr.range)
  310. );
  311. old.addPresentationalDependency(dep);
  312. if (parser.state.module) {
  313. parser.state.module.addError(
  314. new UnsupportedFeatureWarning(
  315. `Cannot statically analyse 'require(…, …)' in line ${
  316. /** @type {SourceLocation} */ (expr.loc).start.line
  317. }`,
  318. /** @type {DependencyLocation} */ (expr.loc)
  319. )
  320. );
  321. }
  322. depBlock = null;
  323. return true;
  324. }
  325. /** @type {AMDRequireDependency} */
  326. (dep).functionBindThis = this.processFunctionArgument(
  327. parser,
  328. expr.arguments[1]
  329. );
  330. if (expr.arguments.length === 3) {
  331. /** @type {AMDRequireDependency} */
  332. (dep).errorCallbackBindThis = this.processFunctionArgument(
  333. parser,
  334. expr.arguments[2]
  335. );
  336. }
  337. } finally {
  338. parser.state.current = old;
  339. if (depBlock) parser.state.current.addBlock(depBlock);
  340. }
  341. return true;
  342. }
  343. }
  344. /**
  345. * @param {DependencyLocation} loc location
  346. * @param {string=} request request
  347. * @returns {AMDRequireDependenciesBlock} AMDRequireDependenciesBlock
  348. */
  349. newRequireDependenciesBlock(loc, request) {
  350. return new AMDRequireDependenciesBlock(loc, request);
  351. }
  352. /**
  353. * @param {Range} outerRange outer range
  354. * @param {Range} arrayRange array range
  355. * @param {Range | null} functionRange function range
  356. * @param {Range | null} errorCallbackRange error callback range
  357. * @returns {AMDRequireDependency} dependency
  358. */
  359. newRequireDependency(
  360. outerRange,
  361. arrayRange,
  362. functionRange,
  363. errorCallbackRange
  364. ) {
  365. return new AMDRequireDependency(
  366. outerRange,
  367. arrayRange,
  368. functionRange,
  369. errorCallbackRange
  370. );
  371. }
  372. /**
  373. * @param {string} request request
  374. * @param {Range=} range range
  375. * @returns {AMDRequireItemDependency} AMDRequireItemDependency
  376. */
  377. newRequireItemDependency(request, range) {
  378. return new AMDRequireItemDependency(request, range);
  379. }
  380. /**
  381. * @param {(string | LocalModuleDependency | AMDRequireItemDependency)[]} depsArray deps array
  382. * @param {Range} range range
  383. * @returns {AMDRequireArrayDependency} AMDRequireArrayDependency
  384. */
  385. newRequireArrayDependency(depsArray, range) {
  386. return new AMDRequireArrayDependency(depsArray, range);
  387. }
  388. }
  389. module.exports = AMDRequireDependenciesBlockParserPlugin;