project/lib/specifications/types/Application.js

  1. import fsPath from "node:path";
  2. import ComponentProject from "../ComponentProject.js";
  3. import {createReader} from "@ui5/fs/resourceFactory";
  4. /**
  5. * Application
  6. *
  7. * @public
  8. * @class
  9. * @alias @ui5/project/specifications/types/Application
  10. * @extends @ui5/project/specifications/ComponentProject
  11. * @hideconstructor
  12. */
  13. class Application extends ComponentProject {
  14. constructor(parameters) {
  15. super(parameters);
  16. this._pManifests = Object.create(null);
  17. this._webappPath = "webapp";
  18. this._isRuntimeNamespaced = false;
  19. }
  20. /* === Attributes === */
  21. /**
  22. * Get the cachebuster signature type configuration of the project
  23. *
  24. * @returns {string} <code>time</code> or <code>hash</code>
  25. */
  26. getCachebusterSignatureType() {
  27. return this._config.builder && this._config.builder.cachebuster &&
  28. this._config.builder.cachebuster.signatureType || "time";
  29. }
  30. /**
  31. * Get the path of the project's source directory. This might not be POSIX-style on some platforms.
  32. *
  33. * @public
  34. * @returns {string} Absolute path to the source directory of the project
  35. */
  36. getSourcePath() {
  37. return fsPath.join(this.getRootPath(), this._webappPath);
  38. }
  39. /* === Resource Access === */
  40. /**
  41. * Get a resource reader for the sources of the project (excluding any test resources)
  42. *
  43. * @param {string[]} excludes List of glob patterns to exclude
  44. * @returns {@ui5/fs/ReaderCollection} Reader collection
  45. */
  46. _getSourceReader(excludes) {
  47. return createReader({
  48. fsBasePath: this.getSourcePath(),
  49. virBasePath: `/resources/${this._namespace}/`,
  50. name: `Source reader for application project ${this.getName()}`,
  51. project: this,
  52. excludes
  53. });
  54. }
  55. _getTestReader() {
  56. return null; // Applications do not have a dedicated test directory
  57. }
  58. /**
  59. * Get a resource reader for the sources of the project (excluding any test resources)
  60. * without a virtual base path
  61. *
  62. * @returns {@ui5/fs/ReaderCollection} Reader collection
  63. */
  64. _getRawSourceReader() {
  65. return createReader({
  66. fsBasePath: this.getSourcePath(),
  67. virBasePath: "/",
  68. name: `Raw source reader for application project ${this.getName()}`,
  69. project: this
  70. });
  71. }
  72. /* === Internals === */
  73. /**
  74. * @private
  75. * @param {object} config Configuration object
  76. */
  77. async _configureAndValidatePaths(config) {
  78. await super._configureAndValidatePaths(config);
  79. if (config.resources && config.resources.configuration &&
  80. config.resources.configuration.paths && config.resources.configuration.paths.webapp) {
  81. this._webappPath = config.resources.configuration.paths.webapp;
  82. }
  83. this._log.verbose(`Path mapping for application project ${this.getName()}:`);
  84. this._log.verbose(` Physical root path: ${this.getRootPath()}`);
  85. this._log.verbose(` Mapped to: ${this._webappPath}`);
  86. if (!(await this._dirExists("/" + this._webappPath))) {
  87. throw new Error(
  88. `Unable to find source directory '${this._webappPath}' in application project ${this.getName()}`);
  89. }
  90. }
  91. /**
  92. * @private
  93. * @param {object} config Configuration object
  94. * @param {object} buildDescription Cache metadata object
  95. */
  96. async _parseConfiguration(config, buildDescription) {
  97. await super._parseConfiguration(config, buildDescription);
  98. if (buildDescription) {
  99. this._namespace = buildDescription.namespace;
  100. return;
  101. }
  102. this._namespace = await this._getNamespace();
  103. }
  104. /**
  105. * Determine application namespace either based on a project`s
  106. * manifest.json or manifest.appdescr_variant (fallback if present)
  107. *
  108. * @returns {string} Namespace of the project
  109. * @throws {Error} if namespace can not be determined
  110. */
  111. async _getNamespace() {
  112. try {
  113. return await this._getNamespaceFromManifestJson();
  114. } catch (manifestJsonError) {
  115. if (manifestJsonError.code !== "ENOENT") {
  116. throw manifestJsonError;
  117. }
  118. // No manifest.json present
  119. // => attempt fallback to manifest.appdescr_variant (typical for App Variants)
  120. try {
  121. return await this._getNamespaceFromManifestAppDescVariant();
  122. } catch (appDescVarError) {
  123. if (appDescVarError.code === "ENOENT") {
  124. // Fallback not possible: No manifest.appdescr_variant present
  125. // => Throw error indicating missing manifest.json
  126. // (do not mention manifest.appdescr_variant since it is only
  127. // relevant for the rather "uncommon" App Variants)
  128. throw new Error(
  129. `Could not find required manifest.json for project ` +
  130. `${this.getName()}: ${manifestJsonError.message}\n\n` +
  131. `If you are about to start a new project, please refer to:\n` +
  132. `https://sap.github.io/ui5-tooling/v3/pages/GettingStarted/#starting-a-new-project`, {
  133. cause: manifestJsonError
  134. });
  135. }
  136. throw appDescVarError;
  137. }
  138. }
  139. }
  140. /**
  141. * Determine application namespace by checking manifest.json.
  142. * Any maven placeholders are resolved from the projects pom.xml
  143. *
  144. * @returns {string} Namespace of the project
  145. * @throws {Error} if namespace can not be determined
  146. */
  147. async _getNamespaceFromManifestJson() {
  148. const manifest = await this._getManifest("/manifest.json");
  149. let appId;
  150. // check for a proper sap.app/id in manifest.json to determine namespace
  151. if (manifest["sap.app"] && manifest["sap.app"].id) {
  152. appId = manifest["sap.app"].id;
  153. } else {
  154. throw new Error(
  155. `No sap.app/id configuration found in manifest.json of project ${this.getName()}`);
  156. }
  157. if (this._hasMavenPlaceholder(appId)) {
  158. try {
  159. appId = await this._resolveMavenPlaceholder(appId);
  160. } catch (err) {
  161. throw new Error(
  162. `Failed to resolve namespace of project ${this.getName()}: ${err.message}`);
  163. }
  164. }
  165. const namespace = appId.replace(/\./g, "/");
  166. this._log.verbose(
  167. `Namespace of project ${this.getName()} is ${namespace} (from manifest.json)`);
  168. return namespace;
  169. }
  170. /**
  171. * Determine application namespace by checking manifest.appdescr_variant.
  172. *
  173. * @returns {string} Namespace of the project
  174. * @throws {Error} if namespace can not be determined
  175. */
  176. async _getNamespaceFromManifestAppDescVariant() {
  177. const manifest = await this._getManifest("/manifest.appdescr_variant");
  178. let appId;
  179. // check for the id property in manifest.appdescr_variant to determine namespace
  180. if (manifest && manifest.id) {
  181. appId = manifest.id;
  182. } else {
  183. throw new Error(
  184. `No "id" property found in manifest.appdescr_variant of project ${this.getName()}`);
  185. }
  186. const namespace = appId.replace(/\./g, "/");
  187. this._log.verbose(
  188. `Namespace of project ${this.getName()} is ${namespace} (from manifest.appdescr_variant)`);
  189. return namespace;
  190. }
  191. /**
  192. * Reads and parses a JSON file with the provided name from the projects source directory
  193. *
  194. * @param {string} filePath Name of the JSON file to read. Typically "manifest.json" or "manifest.appdescr_variant"
  195. * @returns {Promise<object>} resolves with an object containing the content requested manifest file
  196. */
  197. async _getManifest(filePath) {
  198. if (this._pManifests[filePath]) {
  199. return this._pManifests[filePath];
  200. }
  201. return this._pManifests[filePath] = this._getRawSourceReader().byPath(filePath)
  202. .then(async (resource) => {
  203. if (!resource) {
  204. const error = new Error(
  205. `Could not find resource ${filePath} in project ${this.getName()}`);
  206. error.code = "ENOENT"; // "File or directory does not exist"
  207. throw error;
  208. }
  209. return JSON.parse(await resource.getString());
  210. }).catch((err) => {
  211. if (err.code === "ENOENT") {
  212. throw err;
  213. }
  214. throw new Error(
  215. `Failed to read ${filePath} for project ` +
  216. `${this.getName()}: ${err.message}`);
  217. });
  218. }
  219. }
  220. export default Application;