project/lib/specifications/types/Module.js

  1. import fsPath from "node:path";
  2. import Project from "../Project.js";
  3. import * as resourceFactory from "@ui5/fs/resourceFactory";
  4. /**
  5. * Module
  6. *
  7. * @public
  8. * @class
  9. * @alias @ui5/project/specifications/types/Module
  10. * @extends @ui5/project/specifications/Project
  11. * @hideconstructor
  12. */
  13. class Module extends Project {
  14. constructor(parameters) {
  15. super(parameters);
  16. this._paths = null;
  17. this._writer = null;
  18. }
  19. /* === Attributes === */
  20. /**
  21. * Since Modules have multiple source paths, this function always throws with an exception
  22. *
  23. * @public
  24. * @throws {Error} Projects of type module have more than one source path
  25. */
  26. getSourcePath() {
  27. throw new Error(`Projects of type module have more than one source path`);
  28. }
  29. /* === Resource Access === */
  30. /**
  31. * Get a [ReaderCollection]{@link @ui5/fs/ReaderCollection} for accessing all resources of the
  32. * project in the specified "style":
  33. *
  34. * <ul>
  35. * <li><b>buildtime:</b> Resource paths are always prefixed with <code>/resources/</code>
  36. * or <code>/test-resources/</code> followed by the project's namespace.
  37. * Any configured build-excludes are applied</li>
  38. * <li><b>dist:</b> Resource paths always match with what the UI5 runtime expects.
  39. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  40. * structure, while libraries use a "buildtime"-like structure.
  41. * Any configured build-excludes are applied</li>
  42. * <li><b>runtime:</b> Resource paths always match with what the UI5 runtime expects.
  43. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  44. * structure, while libraries use a "buildtime"-like structure.
  45. * This style is typically used for serving resources directly. Therefore, build-excludes are not applied
  46. * <li><b>flat:</b> Resource paths are never prefixed and namespaces are omitted if possible. Note that
  47. * project types like "theme-library", which can have multiple namespaces, can't omit them.
  48. * Any configured build-excludes are applied</li>
  49. * </ul>
  50. *
  51. * If project resources have been changed through the means of a workspace, those changes
  52. * are reflected in the provided reader too.
  53. *
  54. * Resource readers always use POSIX-style paths.
  55. *
  56. * @public
  57. * @param {object} [options]
  58. * @param {string} [options.style=buildtime] Path style to access resources.
  59. * Can be "buildtime", "dist", "runtime" or "flat"
  60. * @returns {@ui5/fs/ReaderCollection} A reader collection instance
  61. */
  62. getReader({style = "buildtime"} = {}) {
  63. // Apply builder excludes to all styles but "runtime"
  64. const excludes = style === "runtime" ? [] : this.getBuilderResourcesExcludes();
  65. const readers = this._paths.map(({name, virBasePath, fsBasePath}) => {
  66. return resourceFactory.createReader({
  67. name,
  68. virBasePath,
  69. fsBasePath,
  70. project: this,
  71. excludes
  72. });
  73. });
  74. if (readers.length === 1) {
  75. return readers[0];
  76. }
  77. const readerCollection = resourceFactory.createReaderCollection({
  78. name: `Reader collection for module project ${this.getName()}`,
  79. readers
  80. });
  81. return resourceFactory.createReaderCollectionPrioritized({
  82. name: `Reader/Writer collection for project ${this.getName()}`,
  83. readers: [this._getWriter(), readerCollection]
  84. });
  85. }
  86. /**
  87. * Get a resource reader/writer for accessing and modifying a project's resources
  88. *
  89. * @public
  90. * @returns {@ui5/fs/ReaderCollection} A reader collection instance
  91. */
  92. getWorkspace() {
  93. const reader = this.getReader();
  94. const writer = this._getWriter();
  95. return resourceFactory.createWorkspace({
  96. reader,
  97. writer
  98. });
  99. }
  100. _getWriter() {
  101. if (!this._writer) {
  102. this._writer = resourceFactory.createAdapter({
  103. virBasePath: "/"
  104. });
  105. }
  106. return this._writer;
  107. }
  108. /* === Internals === */
  109. /**
  110. * @private
  111. * @param {object} config Configuration object
  112. */
  113. async _configureAndValidatePaths(config) {
  114. await super._configureAndValidatePaths(config);
  115. this._log.verbose(`Path mapping for module project ${this.getName()}:`);
  116. this._log.verbose(` Physical root path: ${this.getRootPath()}`);
  117. this._log.verbose(` Mapped to:`);
  118. if (config.resources?.configuration?.paths) {
  119. const pathMappings = Object.entries(config.resources.configuration.paths);
  120. if (this._log.isLevelEnabled("verbose")) {
  121. // Log synchronously before async dir-exists checks
  122. pathMappings.forEach(([virBasePath, relFsPath]) => {
  123. this._log.verbose(` ${virBasePath} => ${relFsPath}`);
  124. });
  125. }
  126. this._paths = await Promise.all(pathMappings.map(async ([virBasePath, relFsPath]) => {
  127. if (!(await this._dirExists("/" + relFsPath))) {
  128. throw new Error(
  129. `Unable to find source directory '${relFsPath}' in module project ${this.getName()}`);
  130. }
  131. return {
  132. name: `'${relFsPath}'' reader for module project ${this.getName()}`,
  133. virBasePath,
  134. fsBasePath: fsPath.join(this.getRootPath(), relFsPath)
  135. };
  136. }));
  137. } else {
  138. this._log.verbose(` / => <project root>`);
  139. if (!(await this._dirExists("/"))) {
  140. throw new Error(
  141. `Unable to find root directory of module project ${this.getName()}`);
  142. }
  143. this._paths = [{
  144. name: `Root reader for module project ${this.getName()}`,
  145. virBasePath: "/",
  146. fsBasePath: this.getRootPath()
  147. }];
  148. }
  149. }
  150. }
  151. export default Module;