project/lib/config/Configuration.js

  1. import path from "node:path";
  2. import os from "node:os";
  3. /**
  4. * Provides basic configuration for @ui5/project.
  5. * Reads/writes configuration from/to ~/.ui5rc
  6. *
  7. * @public
  8. * @class
  9. * @alias @ui5/project/config/Configuration
  10. */
  11. class Configuration {
  12. /**
  13. * A list of all configuration options.
  14. *
  15. * @public
  16. * @static
  17. */
  18. static OPTIONS = [
  19. "mavenSnapshotEndpointUrl",
  20. "ui5DataDir"
  21. ];
  22. #options = new Map();
  23. /**
  24. * @param {object} configuration
  25. * @param {string} [configuration.mavenSnapshotEndpointUrl]
  26. * @param {string} [configuration.ui5DataDir]
  27. */
  28. constructor(configuration) {
  29. // Initialize map with undefined values for every option so that they are
  30. // returned via toJson()
  31. Configuration.OPTIONS.forEach((key) => this.#options.set(key, undefined));
  32. Object.entries(configuration).forEach(([key, value]) => {
  33. if (!Configuration.OPTIONS.includes(key)) {
  34. throw new Error(`Unknown configuration option '${key}'`);
  35. }
  36. this.#options.set(key, value);
  37. });
  38. }
  39. /**
  40. * Maven Repository Snapshot URL.
  41. * Used to download artifacts and packages from Maven's build-snapshots URL.
  42. *
  43. * @public
  44. * @returns {string}
  45. */
  46. getMavenSnapshotEndpointUrl() {
  47. return this.#options.get("mavenSnapshotEndpointUrl");
  48. }
  49. /**
  50. * Configurable directory where the framework artefacts are stored.
  51. *
  52. * @public
  53. * @returns {string}
  54. */
  55. getUi5DataDir() {
  56. return this.#options.get("ui5DataDir");
  57. }
  58. /**
  59. * @public
  60. * @returns {object} The configuration in a JSON format
  61. */
  62. toJson() {
  63. return Object.fromEntries(this.#options);
  64. }
  65. /**
  66. * Creates Configuration from a JSON file
  67. *
  68. * @public
  69. * @static
  70. * @param {string} [filePath="~/.ui5rc"] Path to configuration JSON file
  71. * @returns {Promise<@ui5/project/config/Configuration>} Configuration instance
  72. */
  73. static async fromFile(filePath) {
  74. filePath = filePath || path.resolve(path.join(os.homedir(), ".ui5rc"));
  75. const {default: fs} = await import("graceful-fs");
  76. const {promisify} = await import("node:util");
  77. const readFile = promisify(fs.readFile);
  78. let config;
  79. try {
  80. const fileContent = await readFile(filePath);
  81. if (!fileContent.length) {
  82. config = {};
  83. } else {
  84. config = JSON.parse(fileContent);
  85. }
  86. } catch (err) {
  87. if (err.code === "ENOENT") {
  88. // "File or directory does not exist"
  89. config = {};
  90. } else {
  91. throw new Error(
  92. `Failed to read UI5 Tooling configuration from ${filePath}: ${err.message}`, {
  93. cause: err
  94. });
  95. }
  96. }
  97. return new Configuration(config);
  98. }
  99. /**
  100. * Saves Configuration to a JSON file
  101. *
  102. * @public
  103. * @static
  104. * @param {@ui5/project/config/Configuration} config Configuration to save
  105. * @param {string} [filePath="~/.ui5rc"] Path to configuration JSON file
  106. * @returns {Promise<void>}
  107. */
  108. static async toFile(config, filePath) {
  109. filePath = filePath || path.resolve(path.join(os.homedir(), ".ui5rc"));
  110. const {default: fs} = await import("graceful-fs");
  111. const {promisify} = await import("node:util");
  112. const writeFile = promisify(fs.writeFile);
  113. try {
  114. return writeFile(filePath, JSON.stringify(config.toJson()));
  115. } catch (err) {
  116. throw new Error(
  117. `Failed to write UI5 Tooling configuration to ${filePath}: ${err.message}`, {
  118. cause: err
  119. });
  120. }
  121. }
  122. }
  123. export default Configuration;