project/lib/specifications/ComponentProject.js

  1. import {promisify} from "node:util";
  2. import Project from "./Project.js";
  3. import * as resourceFactory from "@ui5/fs/resourceFactory";
  4. /**
  5. * Subclass for projects potentially containing Components
  6. *
  7. * @public
  8. * @abstract
  9. * @class
  10. * @alias @ui5/project/specifications/ComponentProject
  11. * @extends @ui5/project/specifications/Project
  12. * @hideconstructor
  13. */
  14. class ComponentProject extends Project {
  15. constructor(parameters) {
  16. super(parameters);
  17. if (new.target === ComponentProject) {
  18. throw new TypeError("Class 'ComponentProject' is abstract. Please use one of the 'types' subclasses");
  19. }
  20. this._pPom = null;
  21. this._namespace = null;
  22. this._isRuntimeNamespaced = true;
  23. }
  24. /* === Attributes === */
  25. /**
  26. * Get the project namespace
  27. *
  28. * @public
  29. * @returns {string} Project namespace in slash notation (e.g. <code>my/project/name</code>)
  30. */
  31. getNamespace() {
  32. return this._namespace;
  33. }
  34. /**
  35. * @private
  36. */
  37. getCopyright() {
  38. return this._config.metadata.copyright;
  39. }
  40. /**
  41. * @private
  42. */
  43. getComponentPreloadPaths() {
  44. return this._config.builder && this._config.builder.componentPreload &&
  45. this._config.builder.componentPreload.paths || [];
  46. }
  47. /**
  48. * @private
  49. */
  50. getComponentPreloadNamespaces() {
  51. return this._config.builder && this._config.builder.componentPreload &&
  52. this._config.builder.componentPreload.namespaces || [];
  53. }
  54. /**
  55. * @private
  56. */
  57. getComponentPreloadExcludes() {
  58. return this._config.builder && this._config.builder.componentPreload &&
  59. this._config.builder.componentPreload.excludes || [];
  60. }
  61. /**
  62. * @private
  63. */
  64. getMinificationExcludes() {
  65. return this._config.builder && this._config.builder.minification &&
  66. this._config.builder.minification.excludes || [];
  67. }
  68. /**
  69. * @private
  70. */
  71. getBundles() {
  72. return this._config.builder && this._config.builder.bundles || [];
  73. }
  74. /**
  75. * @private
  76. */
  77. getPropertiesFileSourceEncoding() {
  78. return this._config.resources && this._config.resources.configuration &&
  79. this._config.resources.configuration.propertiesFileSourceEncoding || "UTF-8";
  80. }
  81. /* === Resource Access === */
  82. /**
  83. * Get a [ReaderCollection]{@link @ui5/fs/ReaderCollection} for accessing all resources of the
  84. * project in the specified "style":
  85. *
  86. * <ul>
  87. * <li><b>buildtime:</b> Resource paths are always prefixed with <code>/resources/</code>
  88. * or <code>/test-resources/</code> followed by the project's namespace.
  89. * Any configured build-excludes are applied</li>
  90. * <li><b>dist:</b> Resource paths always match with what the UI5 runtime expects.
  91. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  92. * structure, while libraries use a "buildtime"-like structure.
  93. * Any configured build-excludes are applied</li>
  94. * <li><b>runtime:</b> Resource paths always match with what the UI5 runtime expects.
  95. * This means that paths generally depend on the project type. Applications for example use a "flat"-like
  96. * structure, while libraries use a "buildtime"-like structure.
  97. * This style is typically used for serving resources directly. Therefore, build-excludes are not applied
  98. * <li><b>flat:</b> Resource paths are never prefixed and namespaces are omitted if possible. Note that
  99. * project types like "theme-library", which can have multiple namespaces, can't omit them.
  100. * Any configured build-excludes are applied</li>
  101. * </ul>
  102. *
  103. * If project resources have been changed through the means of a workspace, those changes
  104. * are reflected in the provided reader too.
  105. *
  106. * Resource readers always use POSIX-style paths.
  107. *
  108. * @public
  109. * @param {object} [options]
  110. * @param {string} [options.style=buildtime] Path style to access resources.
  111. * Can be "buildtime", "dist", "runtime" or "flat"
  112. * @returns {@ui5/fs/ReaderCollection} A reader collection instance
  113. */
  114. getReader({style = "buildtime"} = {}) {
  115. // TODO: Additional style 'ABAP' using "sap.platform.abap".uri from manifest.json?
  116. // Apply builder excludes to all styles but "runtime"
  117. const excludes = style === "runtime" ? [] : this.getBuilderResourcesExcludes();
  118. if ((style === "runtime" || style === "dist") && this._isRuntimeNamespaced) {
  119. // If the project's type requires a namespace at runtime, the
  120. // dist- and runtime-style paths are identical to buildtime-style paths
  121. style = "buildtime";
  122. }
  123. let reader = this._getReader(excludes);
  124. switch (style) {
  125. case "buildtime":
  126. break;
  127. case "runtime":
  128. case "dist":
  129. // Use buildtime reader and link it to /
  130. // No test-resources for runtime resource access,
  131. // unless runtime is namespaced
  132. reader = resourceFactory.createFlatReader({
  133. reader,
  134. namespace: this._namespace
  135. });
  136. break;
  137. case "flat":
  138. // Use buildtime reader and link it to /
  139. // No test-resources for runtime resource access,
  140. // unless runtime is namespaced
  141. reader = resourceFactory.createFlatReader({
  142. reader,
  143. namespace: this._namespace
  144. });
  145. break;
  146. default:
  147. throw new Error(`Unknown path mapping style ${style}`);
  148. }
  149. reader = this._addWriter(reader, style);
  150. return reader;
  151. }
  152. /**
  153. * Get a resource reader for the resources of the project
  154. *
  155. * @returns {@ui5/fs/ReaderCollection} Reader collection
  156. */
  157. _getSourceReader() {
  158. throw new Error(`_getSourceReader must be implemented by subclass ${this.constructor.name}`);
  159. }
  160. /**
  161. * Get a resource reader for the test resources of the project
  162. *
  163. * @returns {@ui5/fs/ReaderCollection} Reader collection
  164. */
  165. _getTestReader() {
  166. throw new Error(`_getTestReader must be implemented by subclass ${this.constructor.name}`);
  167. }
  168. /**
  169. * Get a resource reader/writer for accessing and modifying a project's resources
  170. *
  171. * @public
  172. * @returns {@ui5/fs/ReaderCollection} A reader collection instance
  173. */
  174. getWorkspace() {
  175. // Workspace is always of style "buildtime"
  176. // Therefore builder resource-excludes are always to be applied
  177. const excludes = this.getBuilderResourcesExcludes();
  178. return resourceFactory.createWorkspace({
  179. name: `Workspace for project ${this.getName()}`,
  180. reader: this._getReader(excludes),
  181. writer: this._getWriter().collection
  182. });
  183. }
  184. _getWriter() {
  185. if (!this._writers) {
  186. // writer is always of style "buildtime"
  187. const namespaceWriter = resourceFactory.createAdapter({
  188. virBasePath: "/",
  189. project: this
  190. });
  191. const generalWriter = resourceFactory.createAdapter({
  192. virBasePath: "/",
  193. project: this
  194. });
  195. const collection = resourceFactory.createWriterCollection({
  196. name: `Writers for project ${this.getName()}`,
  197. writerMapping: {
  198. [`/resources/${this._namespace}/`]: namespaceWriter,
  199. [`/test-resources/${this._namespace}/`]: namespaceWriter,
  200. [`/`]: generalWriter
  201. }
  202. });
  203. this._writers = {
  204. namespaceWriter,
  205. generalWriter,
  206. collection
  207. };
  208. }
  209. return this._writers;
  210. }
  211. _getReader(excludes) {
  212. let reader = this._getSourceReader(excludes);
  213. const testReader = this._getTestReader(excludes);
  214. if (testReader) {
  215. reader = resourceFactory.createReaderCollection({
  216. name: `Reader collection for project ${this.getName()}`,
  217. readers: [reader, testReader]
  218. });
  219. }
  220. return reader;
  221. }
  222. _addWriter(reader, style) {
  223. const {namespaceWriter, generalWriter} = this._getWriter();
  224. if ((style === "runtime" || style === "dist") && this._isRuntimeNamespaced) {
  225. // If the project's type requires a namespace at runtime, the
  226. // dist- and runtime-style paths are identical to buildtime-style paths
  227. style = "buildtime";
  228. }
  229. const readers = [];
  230. switch (style) {
  231. case "buildtime":
  232. // Writer already uses buildtime style
  233. readers.push(namespaceWriter);
  234. readers.push(generalWriter);
  235. break;
  236. case "runtime":
  237. case "dist":
  238. // Runtime is not namespaced: link namespace to /
  239. readers.push(resourceFactory.createFlatReader({
  240. reader: namespaceWriter,
  241. namespace: this._namespace
  242. }));
  243. // Add general writer as is
  244. readers.push(generalWriter);
  245. break;
  246. case "flat":
  247. // Rewrite paths from "flat" to "buildtime"
  248. readers.push(resourceFactory.createFlatReader({
  249. reader: namespaceWriter,
  250. namespace: this._namespace
  251. }));
  252. // General writer resources can't be flattened, so they are not available
  253. break;
  254. default:
  255. throw new Error(`Unknown path mapping style ${style}`);
  256. }
  257. readers.push(reader);
  258. return resourceFactory.createReaderCollectionPrioritized({
  259. name: `Reader/Writer collection for project ${this.getName()}`,
  260. readers
  261. });
  262. }
  263. /* === Internals === */
  264. /**
  265. * @private
  266. * @param {object} config Configuration object
  267. */
  268. async _parseConfiguration(config) {
  269. await super._parseConfiguration(config);
  270. }
  271. async _getNamespace() {
  272. throw new Error(`_getNamespace must be implemented by subclass ${this.constructor.name}`);
  273. }
  274. /* === Helper === */
  275. /**
  276. * Checks whether a given string contains a maven placeholder.
  277. * E.g. <code>${appId}</code>.
  278. *
  279. * @param {string} value String to check
  280. * @returns {boolean} True if given string contains a maven placeholder
  281. */
  282. _hasMavenPlaceholder(value) {
  283. return !!value.match(/^\$\{(.*)\}$/);
  284. }
  285. /**
  286. * Resolves a maven placeholder in a given string using the projects pom.xml
  287. *
  288. * @param {string} value String containing a maven placeholder
  289. * @returns {Promise<string>} Resolved string
  290. */
  291. async _resolveMavenPlaceholder(value) {
  292. const parts = value && value.match(/^\$\{(.*)\}$/);
  293. if (parts) {
  294. this._log.verbose(
  295. `"${value}" contains a maven placeholder "${parts[1]}". Resolving from projects pom.xml...`);
  296. const pom = await this._getPom();
  297. let mvnValue;
  298. if (pom.project && pom.project.properties && pom.project.properties[parts[1]]) {
  299. mvnValue = pom.project.properties[parts[1]];
  300. } else {
  301. let obj = pom;
  302. parts[1].split(".").forEach((part) => {
  303. obj = obj && obj[part];
  304. });
  305. mvnValue = obj;
  306. }
  307. if (!mvnValue) {
  308. throw new Error(`"${value}" couldn't be resolved from maven property ` +
  309. `"${parts[1]}" of pom.xml of project ${this.getName()}`);
  310. }
  311. return mvnValue;
  312. } else {
  313. throw new Error(`"${value}" is not a maven placeholder`);
  314. }
  315. }
  316. /**
  317. * Reads the projects pom.xml file
  318. *
  319. * @returns {Promise<object>} Resolves with a JSON representation of the content
  320. */
  321. async _getPom() {
  322. if (this._pPom) {
  323. return this._pPom;
  324. }
  325. return this._pPom = this.getRootReader().byPath("/pom.xml")
  326. .then(async (resource) => {
  327. if (!resource) {
  328. throw new Error(
  329. `Could not find pom.xml in project ${this.getName()}`);
  330. }
  331. const content = await resource.getString();
  332. const {
  333. default: xml2js
  334. } = await import("xml2js");
  335. const parser = new xml2js.Parser({
  336. explicitArray: false,
  337. ignoreAttrs: true
  338. });
  339. const readXML = promisify(parser.parseString);
  340. return readXML(content);
  341. }).catch((err) => {
  342. throw new Error(
  343. `Failed to read pom.xml for project ${this.getName()}: ${err.message}`);
  344. });
  345. }
  346. }
  347. export default ComponentProject;