project/lib/build/helpers/TaskUtil.js

  1. import {
  2. createReaderCollection,
  3. createReaderCollectionPrioritized,
  4. createResource,
  5. createFilterReader,
  6. createLinkReader,
  7. createFlatReader
  8. } from "@ui5/fs/resourceFactory";
  9. /**
  10. * Convenience functions for UI5 tasks.
  11. * An instance of this class is passed to every standard UI5 task that requires it.
  12. *
  13. * Custom tasks that define a specification version >= 2.2 will receive an interface
  14. * to an instance of this class when called.
  15. * The set of available functions on that interface depends on the specification
  16. * version defined for the extension.
  17. *
  18. * @public
  19. * @class
  20. * @alias @ui5/project/build/helpers/TaskUtil
  21. * @hideconstructor
  22. */
  23. class TaskUtil {
  24. /**
  25. * Standard Build Tags. See UI5 Tooling
  26. * [RFC 0008]{@link https://github.com/SAP/ui5-tooling/blob/main/rfcs/0008-resource-tagging-during-build.md}
  27. * for details.
  28. *
  29. * @public
  30. * @typedef {object} @ui5/project/build/helpers/TaskUtil~StandardBuildTags
  31. * @property {string} OmitFromBuildResult
  32. * Setting this tag to true will prevent the resource from being written to the build target directory
  33. * @property {string} IsBundle
  34. * This tag identifies resources that contain (i.e. bundle) multiple other resources
  35. * @property {string} IsDebugVariant
  36. * This tag identifies resources that are a debug variant (typically named with a "-dbg" suffix)
  37. * of another resource. This tag is part of the build manifest.
  38. * @property {string} HasDebugVariant
  39. * This tag identifies resources for which a debug variant has been created.
  40. * This tag is part of the build manifest.
  41. */
  42. /**
  43. * Since <code>@ui5/project/build/helpers/ProjectBuildContext</code> is a private class, TaskUtil must not be
  44. * instantiated by modules other than @ui5/project itself.
  45. *
  46. * @param {object} parameters
  47. * @param {@ui5/project/build/helpers/ProjectBuildContext} parameters.projectBuildContext ProjectBuildContext
  48. * @public
  49. */
  50. constructor({projectBuildContext}) {
  51. this._projectBuildContext = projectBuildContext;
  52. /**
  53. * @member {@ui5/project/build/helpers/TaskUtil~StandardBuildTags}
  54. * @public
  55. */
  56. this.STANDARD_TAGS = Object.freeze({
  57. // "Project" tags:
  58. // Will be stored on project instance and are hence part of the build manifest
  59. IsDebugVariant: "ui5:IsDebugVariant",
  60. HasDebugVariant: "ui5:HasDebugVariant",
  61. // "Build" tags:
  62. // Will be stored on the project build context
  63. // They are only available to the build tasks of a single project
  64. OmitFromBuildResult: "ui5:OmitFromBuildResult",
  65. IsBundle: "ui5:IsBundle"
  66. });
  67. }
  68. /**
  69. * Stores a tag with value for a given resource's path. Note that the tag is independent of the supplied
  70. * resource instance. For two resource instances with the same path, the same tag value is returned.
  71. * If the path of a resource is changed, any tag information previously stored for that resource is lost.
  72. *
  73. * </br></br>
  74. * This method is only available to custom task extensions defining
  75. * <b>Specification Version 2.2 and above</b>.
  76. *
  77. * @param {@ui5/fs/Resource} resource Resource-instance the tag should be stored for
  78. * @param {string} tag Name of the tag. Currently only the
  79. * [STANDARD_TAGS]{@link @ui5/project/build/helpers/TaskUtil#STANDARD_TAGS} are allowed
  80. * @param {string|boolean|integer} [value=true] Tag value. Must be primitive
  81. * @public
  82. */
  83. setTag(resource, tag, value) {
  84. if (typeof resource === "string") {
  85. throw new Error("Deprecated parameter: " +
  86. "Since UI5 Tooling 3.0, #setTag requires a resource instance. Strings are no longer accepted");
  87. }
  88. const collection = this._projectBuildContext.getResourceTagCollection(resource, tag);
  89. return collection.setTag(resource, tag, value);
  90. }
  91. /**
  92. * Retrieves the value for a stored tag. If no value is stored, <code>undefined</code> is returned.
  93. *
  94. * </br></br>
  95. * This method is only available to custom task extensions defining
  96. * <b>Specification Version 2.2 and above</b>.
  97. *
  98. * @param {@ui5/fs/Resource} resource Resource-instance the tag should be retrieved for
  99. * @param {string} tag Name of the tag
  100. * @returns {string|boolean|integer|undefined} Tag value for the given resource.
  101. * <code>undefined</code> if no value is available
  102. * @public
  103. */
  104. getTag(resource, tag) {
  105. if (typeof resource === "string") {
  106. throw new Error("Deprecated parameter: " +
  107. "Since UI5 Tooling 3.0, #getTag requires a resource instance. Strings are no longer accepted");
  108. }
  109. const collection = this._projectBuildContext.getResourceTagCollection(resource, tag);
  110. return collection.getTag(resource, tag);
  111. }
  112. /**
  113. * Clears the value of a tag stored for the given resource's path.
  114. * It's like the tag was never set for that resource.
  115. *
  116. * </br></br>
  117. * This method is only available to custom task extensions defining
  118. * <b>Specification Version 2.2 and above</b>.
  119. *
  120. * @param {@ui5/fs/Resource} resource Resource-instance the tag should be cleared for
  121. * @param {string} tag Tag
  122. * @public
  123. */
  124. clearTag(resource, tag) {
  125. if (typeof resource === "string") {
  126. throw new Error("Deprecated parameter: " +
  127. "Since UI5 Tooling 3.0, #clearTag requires a resource instance. Strings are no longer accepted");
  128. }
  129. const collection = this._projectBuildContext.getResourceTagCollection(resource, tag);
  130. return collection.clearTag(resource, tag);
  131. }
  132. /**
  133. * Check whether the project currently being built is the root project.
  134. *
  135. * </br></br>
  136. * This method is only available to custom task extensions defining
  137. * <b>Specification Version 2.2 and above</b>.
  138. *
  139. * @returns {boolean} <code>true</code> if the currently built project is the root project
  140. * @public
  141. */
  142. isRootProject() {
  143. return this._projectBuildContext.isRootProject();
  144. }
  145. /**
  146. * Retrieves a build option defined by its <code>key</code.
  147. * If no option with the given <code>key</code> is stored, <code>undefined</code> is returned.
  148. *
  149. * @param {string} key The option key
  150. * @returns {any|undefined} The build option (or undefined)
  151. * @private
  152. */
  153. getBuildOption(key) {
  154. return this._projectBuildContext.getOption(key);
  155. }
  156. /**
  157. * Callback that is executed once the build has finished
  158. *
  159. * @public
  160. * @callback @ui5/project/build/helpers/TaskUtil~cleanupTaskCallback
  161. * @param {boolean} force Whether the cleanup callback should
  162. * gracefully wait for certain jobs to be completed (<code>false</code>)
  163. * or enforce immediate termination (<code>true</code>)
  164. */
  165. /**
  166. * Register a function that must be executed once the build is finished. This can be used to, for example,
  167. * clean up files temporarily created on the file system. If the callback returns a Promise, it will be waited for.
  168. * It will also be executed in cases where the build has failed or has been aborted.
  169. *
  170. * </br></br>
  171. * This method is only available to custom task extensions defining
  172. * <b>Specification Version 2.2 and above</b>.
  173. *
  174. * @param {@ui5/project/build/helpers/TaskUtil~cleanupTaskCallback} callback Callback to
  175. * register; it will be waited for if it returns a Promise
  176. * @public
  177. */
  178. registerCleanupTask(callback) {
  179. return this._projectBuildContext.registerCleanupTask(callback);
  180. }
  181. /**
  182. * Specification Version-dependent [Project]{@link @ui5/project/specifications/Project} interface.
  183. * For details on individual functions, see [Project]{@link @ui5/project/specifications/Project}
  184. *
  185. * @public
  186. * @typedef {object} @ui5/project/build/helpers/TaskUtil~ProjectInterface
  187. * @property {Function} getType Get the project type
  188. * @property {Function} getName Get the project name
  189. * @property {Function} getVersion Get the project version
  190. * @property {Function} getNamespace Get the project namespace
  191. * @property {Function} getRootReader Get the project rootReader
  192. * @property {Function} getReader Get the project reader
  193. * @property {Function} getRootPath Get the local File System path of the project's root directory
  194. * @property {Function} getSourcePath Get the local File System path of the project's source directory
  195. * @property {Function} getCustomConfiguration Get the project Custom Configuration
  196. * @property {Function} isFrameworkProject Check whether the project is a UI5-Framework project
  197. * @property {Function} getFrameworkName Get the project's framework name configuration
  198. * @property {Function} getFrameworkVersion Get the project's framework version configuration
  199. * @property {Function} getFrameworkDependencies Get the project's framework dependencies configuration
  200. */
  201. /**
  202. * Retrieve a single project from the dependency graph
  203. *
  204. * </br></br>
  205. * This method is only available to custom task extensions defining
  206. * <b>Specification Version 3.0 and above</b>.
  207. *
  208. * @param {string|@ui5/fs/Resource} [projectNameOrResource]
  209. * Name of the project to retrieve or a Resource instance to retrieve the associated project for.
  210. * Defaults to the name of the project currently being built
  211. * @returns {@ui5/project/build/helpers/TaskUtil~ProjectInterface|undefined}
  212. * Specification Version-dependent interface to the Project instance or <code>undefined</code>
  213. * if the project name is unknown or the provided resource is not associated with any project.
  214. * @public
  215. */
  216. getProject(projectNameOrResource) {
  217. if (projectNameOrResource) {
  218. if (typeof projectNameOrResource === "string" || projectNameOrResource instanceof String) {
  219. // A project name has been provided
  220. return this._projectBuildContext.getProject(projectNameOrResource);
  221. } else {
  222. // A Resource instance has been provided
  223. return projectNameOrResource.getProject();
  224. }
  225. }
  226. // No parameter has been provided, default to the project currently being built.
  227. return this._projectBuildContext.getProject();
  228. }
  229. /**
  230. * Retrieve a list of direct dependencies of a given project from the dependency graph.
  231. * Note that this list does not include transitive dependencies.
  232. *
  233. * </br></br>
  234. * This method is only available to custom task extensions defining
  235. * <b>Specification Version 3.0 and above</b>.
  236. *
  237. * @param {string} [projectName] Name of the project to retrieve. Defaults to the project currently being built
  238. * @returns {string[]} Names of all direct dependencies
  239. * @throws {Error} If the requested project is unknown to the graph
  240. * @public
  241. */
  242. getDependencies(projectName) {
  243. return this._projectBuildContext.getDependencies(projectName);
  244. }
  245. /**
  246. * Specification Version-dependent set of [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory}
  247. * functions provided to tasks.
  248. * For details on individual functions, see [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory}
  249. *
  250. * @public
  251. * @typedef {object} @ui5/project/build/helpers/TaskUtil~resourceFactory
  252. * @property {Function} createResource Creates a [Resource]{@link @ui5/fs/Resource}.
  253. * Accepts the same parameters as the [Resource]{@link @ui5/fs/Resource} constructor.
  254. * @property {Function} createReaderCollection Creates a reader collection:
  255. * [ReaderCollection]{@link @ui5/fs/ReaderCollection}
  256. * @property {Function} createReaderCollectionPrioritized Creates a prioritized reader collection:
  257. * [ReaderCollectionPrioritized]{@link @ui5/fs/ReaderCollectionPrioritized}
  258. * @property {Function} createFilterReader
  259. * Create a [Filter-Reader]{@link @ui5/fs/readers/Filter} with the given reader.
  260. * @property {Function} createLinkReader
  261. * Create a [Link-Reader]{@link @ui5/fs/readers/Filter} with the given reader.
  262. * @property {Function} createFlatReader Create a [Link-Reader]{@link @ui5/fs/readers/Link}
  263. * where all requests are prefixed with <code>/resources/<namespace></code>.
  264. */
  265. /**
  266. * Provides limited access to [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory} functions
  267. *
  268. * </br></br>
  269. * This attribute is only available to custom task extensions defining
  270. * <b>Specification Version 3.0 and above</b>.
  271. *
  272. * @type {@ui5/project/build/helpers/TaskUtil~resourceFactory}
  273. * @public
  274. */
  275. resourceFactory = {
  276. createResource,
  277. createReaderCollection,
  278. createReaderCollectionPrioritized,
  279. createFilterReader,
  280. createLinkReader,
  281. createFlatReader,
  282. };
  283. /**
  284. * Get an interface to an instance of this class that only provides those functions
  285. * that are supported by the given custom task extension specification version.
  286. *
  287. * @param {@ui5/project/specifications/SpecificationVersion} specVersion
  288. * SpecVersionComparator instance of the custom task
  289. * @returns {object} An object with bound instance methods supported by the given specification version
  290. */
  291. getInterface(specVersion) {
  292. if (specVersion.lte("2.1")) {
  293. // Tasks defining specVersion <= 2.1 do not have access to any TaskUtil APIs
  294. return undefined;
  295. }
  296. const baseInterface = {
  297. STANDARD_TAGS: this.STANDARD_TAGS,
  298. };
  299. bindFunctions(this, baseInterface, [
  300. "setTag", "clearTag", "getTag", "isRootProject", "registerCleanupTask"
  301. ]);
  302. if (specVersion.gte("3.0")) {
  303. // getProject function, returning an interfaced project instance
  304. baseInterface.getProject = (projectName) => {
  305. const project = this.getProject(projectName);
  306. const baseProjectInterface = {};
  307. bindFunctions(project, baseProjectInterface, [
  308. "getType", "getName", "getVersion", "getNamespace",
  309. "getRootReader", "getReader", "getRootPath", "getSourcePath",
  310. "getCustomConfiguration", "isFrameworkProject", "getFrameworkName",
  311. "getFrameworkVersion", "getFrameworkDependencies"
  312. ]);
  313. return baseProjectInterface;
  314. };
  315. // getDependencies function, returning an array of project names
  316. baseInterface.getDependencies = (projectName) => {
  317. return this.getDependencies(projectName);
  318. };
  319. baseInterface.resourceFactory = Object.create(null);
  320. [
  321. // Once new functions get added, extract this array into a variable
  322. // and enhance based on spec version once new functions get added
  323. "createResource", "createReaderCollection", "createReaderCollectionPrioritized",
  324. "createFilterReader", "createLinkReader", "createFlatReader",
  325. ].forEach((factoryFunction) => {
  326. baseInterface.resourceFactory[factoryFunction] = this.resourceFactory[factoryFunction];
  327. });
  328. }
  329. return baseInterface;
  330. }
  331. }
  332. function bindFunctions(sourceObject, targetObject, funcNames) {
  333. funcNames.forEach((funcName) => {
  334. targetObject[funcName] = sourceObject[funcName].bind(sourceObject);
  335. });
  336. }
  337. export default TaskUtil;