ModuleParseError.js 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. /*
  2. MIT License http://www.opensource.org/licenses/mit-license.php
  3. Author Tobias Koppers @sokra
  4. */
  5. "use strict";
  6. const WebpackError = require("./WebpackError");
  7. const makeSerializable = require("./util/makeSerializable");
  8. /** @typedef {import("./Dependency").SourcePosition} SourcePosition */
  9. /** @typedef {import("./serialization/ObjectMiddleware").ObjectDeserializerContext} ObjectDeserializerContext */
  10. /** @typedef {import("./serialization/ObjectMiddleware").ObjectSerializerContext} ObjectSerializerContext */
  11. const WASM_HEADER = Buffer.from([0x00, 0x61, 0x73, 0x6d]);
  12. class ModuleParseError extends WebpackError {
  13. /**
  14. * @param {string | Buffer} source source code
  15. * @param {Error & { loc?: SourcePosition }} err the parse error
  16. * @param {string[]} loaders the loaders used
  17. * @param {string} type module type
  18. */
  19. constructor(source, err, loaders, type) {
  20. let message = `Module parse failed: ${err && err.message}`;
  21. let loc;
  22. if (
  23. ((Buffer.isBuffer(source) && source.slice(0, 4).equals(WASM_HEADER)) ||
  24. (typeof source === "string" && /^\0asm/.test(source))) &&
  25. !type.startsWith("webassembly")
  26. ) {
  27. message +=
  28. "\nThe module seem to be a WebAssembly module, but module is not flagged as WebAssembly module for webpack.";
  29. message +=
  30. "\nBREAKING CHANGE: Since webpack 5 WebAssembly is not enabled by default and flagged as experimental feature.";
  31. message +=
  32. "\nYou need to enable one of the WebAssembly experiments via 'experiments.asyncWebAssembly: true' (based on async modules) or 'experiments.syncWebAssembly: true' (like webpack 4, deprecated).";
  33. message +=
  34. "\nFor files that transpile to WebAssembly, make sure to set the module type in the 'module.rules' section of the config (e. g. 'type: \"webassembly/async\"').";
  35. } else if (!loaders) {
  36. message +=
  37. "\nYou may need an appropriate loader to handle this file type.";
  38. } else if (loaders.length >= 1) {
  39. message += `\nFile was processed with these loaders:${loaders
  40. .map((loader) => `\n * ${loader}`)
  41. .join("")}`;
  42. message +=
  43. "\nYou may need an additional loader to handle the result of these loaders.";
  44. } else {
  45. message +=
  46. "\nYou may need an appropriate loader to handle this file type, currently no loaders are configured to process this file. See https://webpack.js.org/concepts#loaders";
  47. }
  48. if (
  49. err &&
  50. err.loc &&
  51. typeof err.loc === "object" &&
  52. typeof err.loc.line === "number"
  53. ) {
  54. const lineNumber = err.loc.line;
  55. if (
  56. Buffer.isBuffer(source) ||
  57. // eslint-disable-next-line no-control-regex
  58. /[\0\u0001\u0002\u0003\u0004\u0005\u0006\u0007]/.test(source)
  59. ) {
  60. // binary file
  61. message += "\n(Source code omitted for this binary file)";
  62. } else {
  63. const sourceLines = source.split(/\r?\n/);
  64. const start = Math.max(0, lineNumber - 3);
  65. const linesBefore = sourceLines.slice(start, lineNumber - 1);
  66. const theLine = sourceLines[lineNumber - 1];
  67. const linesAfter = sourceLines.slice(lineNumber, lineNumber + 2);
  68. message += `${linesBefore
  69. .map((l) => `\n| ${l}`)
  70. .join(
  71. ""
  72. )}\n> ${theLine}${linesAfter.map((l) => `\n| ${l}`).join("")}`;
  73. }
  74. loc = { start: err.loc };
  75. } else if (err && err.stack) {
  76. message += `\n${err.stack}`;
  77. }
  78. super(message);
  79. this.name = "ModuleParseError";
  80. this.loc = loc;
  81. this.error = err;
  82. }
  83. /**
  84. * @param {ObjectSerializerContext} context context
  85. */
  86. serialize(context) {
  87. const { write } = context;
  88. write(this.error);
  89. super.serialize(context);
  90. }
  91. /**
  92. * @param {ObjectDeserializerContext} context context
  93. */
  94. deserialize(context) {
  95. const { read } = context;
  96. this.error = read();
  97. super.deserialize(context);
  98. }
  99. }
  100. makeSerializable(ModuleParseError, "webpack/lib/ModuleParseError");
  101. module.exports = ModuleParseError;