project/lib/specifications/types/ThemeLibrary.js

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