project/lib/validation/validator.js

  1. import {fileURLToPath} from "node:url";
  2. import {readFile} from "node:fs/promises";
  3. /**
  4. * @module @ui5/project/validation/validator
  5. * @description A collection of validation related APIs
  6. * @public
  7. */
  8. /**
  9. * @enum {string}
  10. * @private
  11. * @readonly
  12. */
  13. export const SCHEMA_VARIANTS = {
  14. "ui5": "ui5.json",
  15. "ui5-workspace": "ui5-workspace.json"
  16. };
  17. class Validator {
  18. constructor({Ajv, ajvErrors, schemaName}) {
  19. if (!schemaName || !SCHEMA_VARIANTS[schemaName]) {
  20. throw new Error(
  21. `"schemaName" is missing or incorrect. The available schemaName variants are ${Object.keys(
  22. SCHEMA_VARIANTS
  23. ).join(", ")}`
  24. );
  25. }
  26. this._schemaName = SCHEMA_VARIANTS[schemaName];
  27. this.ajv = new Ajv({
  28. allErrors: true,
  29. jsonPointers: true,
  30. loadSchema: Validator.loadSchema
  31. });
  32. ajvErrors(this.ajv);
  33. }
  34. _compileSchema() {
  35. const schemaName = this._schemaName;
  36. if (!this._compiling) {
  37. this._compiling = Promise.resolve().then(async () => {
  38. const schema = await Validator.loadSchema(schemaName);
  39. const validate = await this.ajv.compileAsync(schema);
  40. return validate;
  41. });
  42. }
  43. return this._compiling;
  44. }
  45. async validate({config, project, yaml}) {
  46. const fnValidate = await this._compileSchema();
  47. const valid = fnValidate(config);
  48. if (!valid) {
  49. // Read errors/schema from fnValidate before lazy loading ValidationError module.
  50. // Otherwise they might be cleared already.
  51. const {errors, schema} = fnValidate;
  52. const {default: ValidationError} = await import("./ValidationError.js");
  53. throw new ValidationError({
  54. errors,
  55. schema,
  56. project,
  57. yaml
  58. });
  59. }
  60. }
  61. static async loadSchema(schemaPath) {
  62. const filePath = schemaPath.replace("http://ui5.sap/schema/", "");
  63. const schemaFile = await readFile(
  64. fileURLToPath(new URL(`./schema/${filePath}`, import.meta.url)), {encoding: "utf8"}
  65. );
  66. return JSON.parse(schemaFile);
  67. }
  68. }
  69. const validator = Object.create(null);
  70. async function _validate(schemaName, options) {
  71. if (!validator[schemaName]) {
  72. validator[schemaName] = (async () => {
  73. const {default: Ajv} = await import("ajv");
  74. const {default: ajvErrors} = await import("ajv-errors");
  75. return new Validator({Ajv, ajvErrors, schemaName});
  76. })();
  77. }
  78. const schemaValidator = await validator[schemaName];
  79. await schemaValidator.validate(options);
  80. }
  81. /**
  82. * Validates the given ui5 configuration.
  83. *
  84. * @public
  85. * @function
  86. * @static
  87. * @param {object} options
  88. * @param {object} options.config UI5 Configuration to validate
  89. * @param {object} options.project Project information
  90. * @param {string} options.project.id ID of the project
  91. * @param {object} [options.yaml] YAML information
  92. * @param {string} options.yaml.path Path of the YAML file
  93. * @param {string} options.yaml.source Content of the YAML file
  94. * @param {number} [options.yaml.documentIndex=0] Document index in case the YAML file contains multiple documents
  95. * @throws {@ui5/project/validation/ValidationError}
  96. * Rejects with a {@link @ui5/project/validation/ValidationError ValidationError}
  97. * when the validation fails.
  98. * @returns {Promise<undefined>} Returns a Promise that resolves when the validation succeeds
  99. */
  100. export async function validate(options) {
  101. await _validate("ui5", options);
  102. }
  103. /**
  104. * Validates the given ui5-workspace configuration.
  105. *
  106. * @public
  107. * @function
  108. * @static
  109. * @param {object} options
  110. * @param {object} options.config ui5-workspace Configuration to validate
  111. * @param {object} [options.yaml] YAML information
  112. * @param {string} options.yaml.path Path of the YAML file
  113. * @param {string} options.yaml.source Content of the YAML file
  114. * @param {number} [options.yaml.documentIndex=0] Document index in case the YAML file contains multiple documents
  115. * @throws {@ui5/project/validation/ValidationError}
  116. * Rejects with a {@link @ui5/project/validation/ValidationError ValidationError}
  117. * when the validation fails.
  118. * @returns {Promise<undefined>} Returns a Promise that resolves when the validation succeeds
  119. */
  120. export async function validateWorkspace(options) {
  121. await _validate("ui5-workspace", options);
  122. }
  123. export {
  124. /**
  125. * For testing only!
  126. *
  127. * @private
  128. */
  129. Validator as _Validator
  130. };