ModuleParseError.js 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  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. /[\0\u0001\u0002\u0003\u0004\u0005\u0006\u0007]/.test(source)
  58. ) {
  59. // binary file
  60. message += "\n(Source code omitted for this binary file)";
  61. } else {
  62. const sourceLines = source.split(/\r?\n/);
  63. const start = Math.max(0, lineNumber - 3);
  64. const linesBefore = sourceLines.slice(start, lineNumber - 1);
  65. const theLine = sourceLines[lineNumber - 1];
  66. const linesAfter = sourceLines.slice(lineNumber, lineNumber + 2);
  67. message += `${linesBefore
  68. .map(l => `\n| ${l}`)
  69. .join("")}\n> ${theLine}${linesAfter.map(l => `\n| ${l}`).join("")}`;
  70. }
  71. loc = { start: err.loc };
  72. } else if (err && err.stack) {
  73. message += `\n${err.stack}`;
  74. }
  75. super(message);
  76. this.name = "ModuleParseError";
  77. this.loc = loc;
  78. this.error = err;
  79. }
  80. /**
  81. * @param {ObjectSerializerContext} context context
  82. */
  83. serialize(context) {
  84. const { write } = context;
  85. write(this.error);
  86. super.serialize(context);
  87. }
  88. /**
  89. * @param {ObjectDeserializerContext} context context
  90. */
  91. deserialize(context) {
  92. const { read } = context;
  93. this.error = read();
  94. super.deserialize(context);
  95. }
  96. }
  97. makeSerializable(ModuleParseError, "webpack/lib/ModuleParseError");
  98. module.exports = ModuleParseError;