project/lib/validation/ValidationError.js

  1. import chalk from "chalk";
  2. import escapeStringRegExp from "escape-string-regexp";
  3. /**
  4. * Error class for validation of project configuration.
  5. *
  6. * @public
  7. * @class
  8. * @alias @ui5/project/validation/ValidationError
  9. * @extends Error
  10. * @hideconstructor
  11. */
  12. class ValidationError extends Error {
  13. constructor({errors, project, yaml}) {
  14. super();
  15. /**
  16. * ValidationError
  17. *
  18. * @constant
  19. * @default
  20. * @type {string}
  21. * @readonly
  22. * @public
  23. */
  24. this.name = "ValidationError";
  25. this.project = project;
  26. this.yaml = yaml;
  27. this.errors = ValidationError.filterErrors(errors);
  28. /**
  29. * Formatted error message
  30. *
  31. * @type {string}
  32. * @readonly
  33. * @public
  34. */
  35. this.message = this.formatErrors();
  36. Error.captureStackTrace(this, this.constructor);
  37. }
  38. formatErrors() {
  39. let separator = "\n\n";
  40. if (process.stdout.isTTY) {
  41. // Add a horizontal separator line between errors in case a terminal is used
  42. separator += chalk.grey.dim("\u2500".repeat(process.stdout.columns || 80));
  43. }
  44. separator += "\n\n";
  45. let message;
  46. if (this.project) { // ui5-workspace.yaml is project independent, so in that case, no project is available
  47. message = chalk.red(`Invalid ui5.yaml configuration for project ${this.project.id}`) + "\n\n";
  48. } else {
  49. message = chalk.red(`Invalid workspace configuration.`) + "\n\n";
  50. }
  51. message += this.errors.map((error) => {
  52. return this.formatError(error);
  53. }).join(separator);
  54. return message;
  55. }
  56. formatError(error) {
  57. let errorMessage = ValidationError.formatMessage(error);
  58. if (this.yaml && this.yaml.path && this.yaml.source) {
  59. const yamlExtract = ValidationError.getYamlExtract({error, yaml: this.yaml});
  60. const errorLines = errorMessage.split("\n");
  61. errorLines.splice(1, 0, "\n" + yamlExtract);
  62. errorMessage = errorLines.join("\n");
  63. }
  64. return errorMessage;
  65. }
  66. static formatMessage(error) {
  67. if (error.keyword === "errorMessage") {
  68. return error.message;
  69. }
  70. let message = "Configuration ";
  71. if (error.dataPath) {
  72. message += chalk.underline(chalk.red(error.dataPath.substr(1))) + " ";
  73. }
  74. switch (error.keyword) {
  75. case "additionalProperties":
  76. message += `property ${error.params.additionalProperty} must not be provided here`;
  77. break;
  78. case "type":
  79. message += `must be of type '${error.params.type}'`;
  80. break;
  81. case "required":
  82. message += `must have required property '${error.params.missingProperty}'`;
  83. break;
  84. case "enum":
  85. message += "must be equal to one of the allowed values\n";
  86. message += "Allowed values: " + error.params.allowedValues.join(", ");
  87. break;
  88. default:
  89. message += error.message;
  90. }
  91. return message;
  92. }
  93. static _findDuplicateError(error, errorIndex, errors) {
  94. const foundIndex = errors.findIndex(($) => {
  95. if ($.dataPath !== error.dataPath) {
  96. return false;
  97. } else if ($.keyword !== error.keyword) {
  98. return false;
  99. } else if (JSON.stringify($.params) !== JSON.stringify(error.params)) {
  100. return false;
  101. } else {
  102. return true;
  103. }
  104. });
  105. return foundIndex !== errorIndex;
  106. }
  107. static filterErrors(allErrors) {
  108. return allErrors.filter((error, i, errors) => {
  109. if (error.keyword === "if" || error.keyword === "oneOf") {
  110. return false;
  111. }
  112. return !ValidationError._findDuplicateError(error, i, errors);
  113. });
  114. }
  115. static analyzeYamlError({error, yaml}) {
  116. if (error.dataPath === "" && error.keyword === "required") {
  117. // There is no line/column for a missing required property on root level
  118. return {line: -1, column: -1};
  119. }
  120. // Skip leading /
  121. const objectPath = error.dataPath.substr(1).split("/");
  122. if (error.keyword === "additionalProperties") {
  123. objectPath.push(error.params.additionalProperty);
  124. }
  125. let currentSubstring;
  126. let currentIndex;
  127. if (yaml.documentIndex) {
  128. const matchDocumentSeparator = /^---/gm;
  129. let currentDocumentIndex = 0;
  130. let document;
  131. while ((document = matchDocumentSeparator.exec(yaml.source)) !== null) {
  132. // If the first separator is not at the beginning of the file
  133. // we are already at document index 1
  134. // Using String#trim() to remove any whitespace characters
  135. if (currentDocumentIndex === 0 && yaml.source.substring(0, document.index).trim().length > 0) {
  136. currentDocumentIndex = 1;
  137. }
  138. if (currentDocumentIndex === yaml.documentIndex) {
  139. currentIndex = document.index;
  140. currentSubstring = yaml.source.substring(currentIndex);
  141. break;
  142. }
  143. currentDocumentIndex++;
  144. }
  145. // Document could not be found
  146. if (!currentSubstring) {
  147. return {line: -1, column: -1};
  148. }
  149. } else {
  150. // In case of index 0 or no index, use whole source
  151. currentIndex = 0;
  152. currentSubstring = yaml.source;
  153. }
  154. const matchArrayElementIndentation = /([ ]*)-/;
  155. for (let i = 0; i < objectPath.length; i++) {
  156. const property = objectPath[i];
  157. let newIndex;
  158. if (isNaN(property)) {
  159. // Try to find a property
  160. // Creating a regular expression that matches the property name a line
  161. // except for comments, indicated by a hash sign "#".
  162. const propertyRegExp = new RegExp(`^[^#]*?${escapeStringRegExp(property)}`, "m");
  163. const propertyMatch = propertyRegExp.exec(currentSubstring);
  164. if (!propertyMatch) {
  165. return {line: -1, column: -1};
  166. }
  167. newIndex = propertyMatch.index + propertyMatch[0].length;
  168. } else {
  169. // Try to find the right index within an array definition.
  170. // This currently only works for arrays defined with "-" in multiple lines.
  171. // Arrays using square brackets are not supported.
  172. const matchArrayElement = /(^|\r?\n)([ ]*-[^\r\n]*)/g;
  173. const arrayIndex = parseInt(property);
  174. let a = 0;
  175. let firstIndentation = -1;
  176. let match;
  177. while ((match = matchArrayElement.exec(currentSubstring)) !== null) {
  178. const indentationMatch = match[2].match(matchArrayElementIndentation);
  179. if (!indentationMatch) {
  180. return {line: -1, column: -1};
  181. }
  182. const currentIndentation = indentationMatch[1].length;
  183. if (firstIndentation === -1) {
  184. firstIndentation = currentIndentation;
  185. } else if (currentIndentation !== firstIndentation) {
  186. continue;
  187. }
  188. if (a === arrayIndex) {
  189. // match[1] might be a line-break
  190. newIndex = match.index + match[1].length + currentIndentation;
  191. break;
  192. }
  193. a++;
  194. }
  195. if (!newIndex) {
  196. // Could not find array element
  197. return {line: -1, column: -1};
  198. }
  199. }
  200. currentIndex += newIndex;
  201. currentSubstring = yaml.source.substring(currentIndex);
  202. }
  203. const linesUntilMatch = yaml.source.substring(0, currentIndex).split(/\r?\n/);
  204. const line = linesUntilMatch.length;
  205. let column = linesUntilMatch[line - 1].length + 1;
  206. const lastPathSegment = objectPath[objectPath.length - 1];
  207. if (isNaN(lastPathSegment)) {
  208. column -= lastPathSegment.length;
  209. }
  210. return {
  211. line,
  212. column
  213. };
  214. }
  215. static getSourceExtract(yamlSource, line, column) {
  216. let source = "";
  217. const lines = yamlSource.split(/\r?\n/);
  218. // Using line numbers instead of array indices
  219. const startLine = Math.max(line - 2, 1);
  220. const endLine = Math.min(line, lines.length);
  221. const padLength = String(endLine).length;
  222. for (let currentLine = startLine; currentLine <= endLine; currentLine++) {
  223. const currentLineContent = lines[currentLine - 1];
  224. let string = chalk.gray(
  225. String(currentLine).padStart(padLength, " ") + ":"
  226. ) + " " + currentLineContent + "\n";
  227. if (currentLine === line) {
  228. string = chalk.bgRed(string);
  229. }
  230. source += string;
  231. }
  232. source += " ".repeat(column + padLength + 1) + chalk.red("^");
  233. return source;
  234. }
  235. static getYamlExtract({error, yaml}) {
  236. const {line, column} = ValidationError.analyzeYamlError({error, yaml});
  237. if (line !== -1 && column !== -1) {
  238. return chalk.grey(yaml.path + ":" + line) +
  239. "\n\n" + ValidationError.getSourceExtract(yaml.source, line, column);
  240. } else {
  241. return chalk.grey(yaml.path) + "\n";
  242. }
  243. }
  244. }
  245. export default ValidationError;