formatWebpackMessages.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. /**
  2. * Copyright (c) 2015-present, Facebook, Inc.
  3. *
  4. * This source code is licensed under the MIT license found in the
  5. * LICENSE file in the root directory of this source tree.
  6. */
  7. 'use strict';
  8. const friendlySyntaxErrorLabel = 'Syntax error:';
  9. function isLikelyASyntaxError(message) {
  10. return message.indexOf(friendlySyntaxErrorLabel) !== -1;
  11. }
  12. // Cleans up webpack error messages.
  13. function formatMessage(message) {
  14. let lines = [];
  15. if (typeof message === 'string') {
  16. lines = message.split('\n');
  17. } else if ('message' in message) {
  18. lines = message['message'].split('\n');
  19. } else if (Array.isArray(message)) {
  20. message.forEach(message => {
  21. if ('message' in message) {
  22. lines = message['message'].split('\n');
  23. }
  24. });
  25. }
  26. // Strip webpack-added headers off errors/warnings
  27. // https://github.com/webpack/webpack/blob/master/lib/ModuleError.js
  28. lines = lines.filter(line => !/Module [A-z ]+\(from/.test(line));
  29. // Transform parsing error into syntax error
  30. // TODO: move this to our ESLint formatter?
  31. lines = lines.map(line => {
  32. const parsingError = /Line (\d+):(?:(\d+):)?\s*Parsing error: (.+)$/.exec(
  33. line
  34. );
  35. if (!parsingError) {
  36. return line;
  37. }
  38. const [, errorLine, errorColumn, errorMessage] = parsingError;
  39. return `${friendlySyntaxErrorLabel} ${errorMessage} (${errorLine}:${errorColumn})`;
  40. });
  41. message = lines.join('\n');
  42. // Smoosh syntax errors (commonly found in CSS)
  43. message = message.replace(
  44. /SyntaxError\s+\((\d+):(\d+)\)\s*(.+?)\n/g,
  45. `${friendlySyntaxErrorLabel} $3 ($1:$2)\n`
  46. );
  47. // Clean up export errors
  48. message = message.replace(
  49. /^.*export '(.+?)' was not found in '(.+?)'.*$/gm,
  50. `Attempted import error: '$1' is not exported from '$2'.`
  51. );
  52. message = message.replace(
  53. /^.*export 'default' \(imported as '(.+?)'\) was not found in '(.+?)'.*$/gm,
  54. `Attempted import error: '$2' does not contain a default export (imported as '$1').`
  55. );
  56. message = message.replace(
  57. /^.*export '(.+?)' \(imported as '(.+?)'\) was not found in '(.+?)'.*$/gm,
  58. `Attempted import error: '$1' is not exported from '$3' (imported as '$2').`
  59. );
  60. lines = message.split('\n');
  61. // Remove leading newline
  62. if (lines.length > 2 && lines[1].trim() === '') {
  63. lines.splice(1, 1);
  64. }
  65. // Clean up file name
  66. lines[0] = lines[0].replace(/^(.*) \d+:\d+-\d+$/, '$1');
  67. // Cleans up verbose "module not found" messages for files and packages.
  68. if (lines[1] && lines[1].indexOf('Module not found: ') === 0) {
  69. lines = [
  70. lines[0],
  71. lines[1]
  72. .replace('Error: ', '')
  73. .replace('Module not found: Cannot find file:', 'Cannot find file:'),
  74. ];
  75. }
  76. // Add helpful message for users trying to use Sass for the first time
  77. if (lines[1] && lines[1].match(/Cannot find module.+sass/)) {
  78. lines[1] = 'To import Sass files, you first need to install sass.\n';
  79. lines[1] +=
  80. 'Run `npm install sass` or `yarn add sass` inside your workspace.';
  81. }
  82. message = lines.join('\n');
  83. // Internal stacks are generally useless so we strip them... with the
  84. // exception of stacks containing `webpack:` because they're normally
  85. // from user code generated by webpack. For more information see
  86. // https://github.com/facebook/create-react-app/pull/1050
  87. message = message.replace(
  88. /^\s*at\s((?!webpack:).)*:\d+:\d+[\s)]*(\n|$)/gm,
  89. ''
  90. ); // at ... ...:x:y
  91. message = message.replace(/^\s*at\s<anonymous>(\n|$)/gm, ''); // at <anonymous>
  92. lines = message.split('\n');
  93. // Remove duplicated newlines
  94. lines = lines.filter(
  95. (line, index, arr) =>
  96. index === 0 || line.trim() !== '' || line.trim() !== arr[index - 1].trim()
  97. );
  98. // Reassemble the message
  99. message = lines.join('\n');
  100. return message.trim();
  101. }
  102. function formatWebpackMessages(json) {
  103. const formattedErrors = json.errors.map(formatMessage);
  104. const formattedWarnings = json.warnings.map(formatMessage);
  105. const result = { errors: formattedErrors, warnings: formattedWarnings };
  106. if (result.errors.some(isLikelyASyntaxError)) {
  107. // If there are any syntax errors, show just them.
  108. result.errors = result.errors.filter(isLikelyASyntaxError);
  109. }
  110. return result;
  111. }
  112. module.exports = formatWebpackMessages;