project/lib/specifications/Project.js

  1. import Specification from "./Specification.js";
  2. import ResourceTagCollection from "@ui5/fs/internal/ResourceTagCollection";
  3. /**
  4. * Project
  5. *
  6. * @public
  7. * @abstract
  8. * @class
  9. * @alias @ui5/project/specifications/Project
  10. * @extends @ui5/project/specifications/Specification
  11. * @hideconstructor
  12. */
  13. class Project extends Specification {
  14. constructor(parameters) {
  15. super(parameters);
  16. if (new.target === Project) {
  17. throw new TypeError("Class 'Project' is abstract. Please use one of the 'types' subclasses");
  18. }
  19. this._resourceTagCollection = null;
  20. }
  21. /**
  22. * @param {object} parameters Specification parameters
  23. * @param {string} parameters.id Unique ID
  24. * @param {string} parameters.version Version
  25. * @param {string} parameters.modulePath File System path to access resources
  26. * @param {object} parameters.configuration Configuration object
  27. * @param {object} [parameters.buildManifest] Build metadata object
  28. */
  29. async init(parameters) {
  30. await super.init(parameters);
  31. this._buildManifest = parameters.buildManifest;
  32. await this._configureAndValidatePaths(this._config);
  33. await this._parseConfiguration(this._config, this._buildManifest);
  34. return this;
  35. }
  36. /* === Attributes === */
  37. /**
  38. * Get the project namespace. Returns `null` for projects that have none or multiple namespaces,
  39. * for example Modules or Theme Libraries.
  40. *
  41. * @public
  42. * @returns {string|null} Project namespace in slash notation (e.g. <code>my/project/name</code>) or null
  43. */
  44. getNamespace() {
  45. // Default namespace for general Projects:
  46. // Their resources should be structured with globally unique paths, hence their namespace is undefined
  47. return null;
  48. }
  49. /**
  50. * Check whether the project is a UI5-Framework project
  51. *
  52. * @public
  53. * @returns {boolean} True if the project is a framework project
  54. */
  55. isFrameworkProject() {
  56. const id = this.getId();
  57. return id.startsWith("@openui5/") || id.startsWith("@sapui5/");
  58. }
  59. /**
  60. * Get the project's customConfiguration
  61. *
  62. * @public
  63. * @returns {object} Custom Configuration
  64. */
  65. getCustomConfiguration() {
  66. return this._config.customConfiguration;
  67. }
  68. /**
  69. * Get the path of the project's source directory. This might not be POSIX-style on some platforms.
  70. * Projects with multiple source paths will throw an error. For example Modules.
  71. *
  72. * @public
  73. * @returns {string} Absolute path to the source directory of the project
  74. * @throws {Error} In case a project has multiple source directories
  75. */
  76. getSourcePath() {
  77. throw new Error(`getSourcePath must be implemented by subclass ${this.constructor.name}`);
  78. }
  79. /**
  80. * Get the project's framework name configuration
  81. *
  82. * @public
  83. * @returns {string} Framework name configuration, either <code>OpenUI5</code> or <code>SAPUI5</code>
  84. */
  85. getFrameworkName() {
  86. return this._config.framework?.name;
  87. }
  88. /**
  89. * Get the project's framework version configuration
  90. *
  91. * @public
  92. * @returns {string} Framework version configuration, e.g <code>1.110.0</code>
  93. */
  94. getFrameworkVersion() {
  95. return this._config.framework?.version;
  96. }
  97. /**
  98. * Framework dependency entry of the project configuration.
  99. * Also see [Framework Configuration: Dependencies]{@link https://sap.github.io/ui5-tooling/stable/pages/Configuration/#dependencies}
  100. *
  101. * @public
  102. * @typedef {object} @ui5/project/specifications/Project~FrameworkDependency
  103. * @property {string} name Name of the framework library. For example <code>sap.ui.core</code>
  104. * @property {boolean} development Whether the dependency is meant for development purposes only
  105. * @property {boolean} optional Whether the dependency should be treated as optional
  106. */
  107. /**
  108. * Get the project's framework dependencies configuration
  109. *
  110. * @public
  111. * @returns {@ui5/project/specifications/Project~FrameworkDependency[]} Framework dependencies configuration
  112. */
  113. getFrameworkDependencies() {
  114. return this._config.framework?.libraries || [];
  115. }
  116. /**
  117. * Get the project's deprecated configuration
  118. *
  119. * @private
  120. * @returns {boolean} True if the project is flagged as deprecated
  121. */
  122. isDeprecated() {
  123. return !!this._config.metadata.deprecated;
  124. }
  125. /**
  126. * Get the project's sapInternal configuration
  127. *
  128. * @private
  129. * @returns {boolean} True if the project is flagged as SAP-internal
  130. */
  131. isSapInternal() {
  132. return !!this._config.metadata.sapInternal;
  133. }
  134. /**
  135. * Get the project's allowSapInternal configuration
  136. *
  137. * @private
  138. * @returns {boolean} True if the project allows for using SAP-internal projects
  139. */
  140. getAllowSapInternal() {
  141. return !!this._config.metadata.allowSapInternal;
  142. }
  143. /**
  144. * Get the project's builderResourcesExcludes configuration
  145. *
  146. * @private
  147. * @returns {string[]} BuilderResourcesExcludes configuration
  148. */
  149. getBuilderResourcesExcludes() {
  150. return this._config.builder?.resources?.excludes || [];
  151. }
  152. /**
  153. * Get the project's customTasks configuration
  154. *
  155. * @private
  156. * @returns {object[]} CustomTasks configuration
  157. */
  158. getCustomTasks() {
  159. return this._config.builder?.customTasks || [];
  160. }
  161. /**
  162. * Get the project's customMiddleware configuration
  163. *
  164. * @private
  165. * @returns {object[]} CustomMiddleware configuration
  166. */
  167. getCustomMiddleware() {
  168. return this._config.server?.customMiddleware || [];
  169. }
  170. /**
  171. * Get the project's serverSettings configuration
  172. *
  173. * @private
  174. * @returns {object} ServerSettings configuration
  175. */
  176. getServerSettings() {
  177. return this._config.server?.settings;
  178. }
  179. /**
  180. * Get the project's builderSettings configuration
  181. *
  182. * @private
  183. * @returns {object} BuilderSettings configuration
  184. */
  185. getBuilderSettings() {
  186. return this._config.builder?.settings;
  187. }
  188. /**
  189. * Get the project's buildManifest configuration
  190. *
  191. * @private
  192. * @returns {object|null} BuildManifest configuration or null if none is available
  193. */
  194. getBuildManifest() {
  195. return this._buildManifest || null;
  196. }
  197. /* === Resource Access === */
  198. /**
  199. * Get a [ReaderCollection]{@link @ui5/fs/ReaderCollection} for accessing all resources of the
  200. * project in the specified "style":
  201. *
  202. * <ul>
  203. * <li><b>buildtime:</b> Resource paths are always prefixed with <code>/resources/</code>
  204. * or <code>/test-resources/</code> followed by the project's namespace.
  205. * Any configured build-excludes are applied</li>
  206. * <li><b>dist:</b> Resource paths always match with what the UI5 runtime expects.
  207. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  208. * structure, while libraries use a "buildtime"-like structure.
  209. * Any configured build-excludes are applied</li>
  210. * <li><b>runtime:</b> Resource paths always match with what the UI5 runtime expects.
  211. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  212. * structure, while libraries use a "buildtime"-like structure.
  213. * This style is typically used for serving resources directly. Therefore, build-excludes are not applied
  214. * <li><b>flat:</b> Resource paths are never prefixed and namespaces are omitted if possible. Note that
  215. * project types like "theme-library", which can have multiple namespaces, can't omit them.
  216. * Any configured build-excludes are applied</li>
  217. * </ul>
  218. *
  219. * Resource readers always use POSIX-style paths.
  220. *
  221. * @public
  222. * @param {object} [options]
  223. * @param {string} [options.style=buildtime] Path style to access resources.
  224. * Can be "buildtime", "dist", "runtime" or "flat"
  225. * @returns {@ui5/fs/ReaderCollection} Reader collection allowing access to all resources of the project
  226. */
  227. getReader(options) {
  228. throw new Error(`getReader must be implemented by subclass ${this.constructor.name}`);
  229. }
  230. getResourceTagCollection() {
  231. if (!this._resourceTagCollection) {
  232. this._resourceTagCollection = new ResourceTagCollection({
  233. allowedTags: ["ui5:IsDebugVariant", "ui5:HasDebugVariant"],
  234. allowedNamespaces: ["project"],
  235. tags: this.getBuildManifest()?.tags
  236. });
  237. }
  238. return this._resourceTagCollection;
  239. }
  240. /**
  241. * Get a [DuplexCollection]{@link @ui5/fs/DuplexCollection} for accessing and modifying a
  242. * project's resources. This is always of style <code>buildtime</code>.
  243. *
  244. * @public
  245. * @returns {@ui5/fs/DuplexCollection} DuplexCollection
  246. */
  247. getWorkspace() {
  248. throw new Error(`getWorkspace must be implemented by subclass ${this.constructor.name}`);
  249. }
  250. /* === Internals === */
  251. /**
  252. * @private
  253. * @param {object} config Configuration object
  254. */
  255. async _configureAndValidatePaths(config) {}
  256. /**
  257. * @private
  258. * @param {object} config Configuration object
  259. */
  260. async _parseConfiguration(config) {}
  261. }
  262. export default Project;