builder/lib/tasks/bundlers/generateFlexChangesBundle.js

  1. import {getLogger} from "@ui5/logger";
  2. const log = getLogger("builder:tasks:bundlers:generateFlexChangesBundle");
  3. import flexChangesBundler from "../../processors/bundlers/flexChangesBundler.js";
  4. import semver from "semver";
  5. /**
  6. * @public
  7. * @module @ui5/builder/tasks/bundlers/generateFlexChangesBundle
  8. */
  9. /* eslint "jsdoc/check-param-names": ["error", {"disableExtraPropertyReporting":true}] */
  10. /**
  11. * Task to create changesBundle.json file containing all changes stored in the /changes folder for easier consumption
  12. * at runtime.
  13. * If a change bundle is created, "sap.ui.fl" is added as a dependency to the manifest.json if not already present -
  14. * if the dependency is already listed but lazy-loaded, lazy loading is disabled.
  15. * If minUI5Version >= 1.73 flexibility-bundle.json will be create.
  16. * If there are control variants and minUI5Version < 1.73 build will break and throw an error.
  17. *
  18. * @public
  19. * @function default
  20. * @static
  21. *
  22. * @param {object} parameters Parameters
  23. * @param {@ui5/fs/DuplexCollection} parameters.workspace DuplexCollection to read and write files
  24. * @param {@ui5/project/build/helpers/TaskUtil|object} [parameters.taskUtil] TaskUtil
  25. * @param {object} [parameters.options] Options
  26. * @param {string} [parameters.options.projectNamespace] Project Namespace
  27. * @returns {Promise<undefined>} Promise resolving with <code>undefined</code> once data has been written
  28. */
  29. export default async function({workspace, taskUtil, options = {}}) {
  30. // Backward compatibility: "namespace" option got renamed to "projectNamespace"
  31. const namespace = options.projectNamespace || options.namespace;
  32. // Use the given namespace if available, otherwise use no namespace
  33. // (e.g. in case no manifest.json is present)
  34. let pathPrefix = "";
  35. if (namespace) {
  36. pathPrefix = `/resources/${namespace}`;
  37. }
  38. function updateJson(data) {
  39. // ensure the existence of the libs section in the dependencies
  40. data["sap.ui5"] = data["sap.ui5"] || {};
  41. data["sap.ui5"].dependencies = data["sap.ui5"].dependencies || {};
  42. const mLibs = data["sap.ui5"].dependencies.libs = data["sap.ui5"].dependencies.libs || {};
  43. if (mLibs["sap.ui.fl"]) {
  44. log.verbose("sap.ui.fl found in manifest.json");
  45. if (mLibs["sap.ui.fl"].lazy) {
  46. log.verbose("sap.ui.fl 'lazy' attribute found in manifest.json, setting it to false...");
  47. mLibs["sap.ui.fl"].lazy = false;
  48. }
  49. } else {
  50. log.verbose("sap.ui.fl not found in manifest.json, inserting it...");
  51. mLibs["sap.ui.fl"] = {};
  52. }
  53. }
  54. async function updateFLdependency() {
  55. const manifestResource = await workspace.byPath(`${pathPrefix}/manifest.json`);
  56. const manifestContent = JSON.parse(await manifestResource.getString());
  57. updateJson(manifestContent);
  58. manifestResource.setString(JSON.stringify(manifestContent, null, "\t"));
  59. await workspace.write(manifestResource);
  60. }
  61. async function readManifestMinUI5Version() {
  62. const manifestResource = await workspace.byPath(`${pathPrefix}/manifest.json`);
  63. const manifestContent = JSON.parse(await manifestResource.getString());
  64. manifestContent["sap.ui5"] = manifestContent["sap.ui5"] || {};
  65. manifestContent["sap.ui5"].dependencies = manifestContent["sap.ui5"].dependencies || {};
  66. if (!Array.isArray(manifestContent["sap.ui5"].dependencies.minUI5Version)) {
  67. manifestContent["sap.ui5"].dependencies.minUI5Version =
  68. [manifestContent["sap.ui5"].dependencies.minUI5Version] || [""];
  69. }
  70. return manifestContent["sap.ui5"].dependencies.minUI5Version;
  71. }
  72. log.verbose("Collecting flexibility changes");
  73. const allResources = await workspace.byGlob(
  74. `${pathPrefix}/changes/*.{change,variant,ctrl_variant,ctrl_variant_change,ctrl_variant_management_change}`);
  75. if (allResources.length > 0) {
  76. const versionArray = await readManifestMinUI5Version();
  77. const versions = versionArray.map((version) => semver.coerce(version));
  78. const versionsAllSuitableForFlexBundle = versions.every((version) => semver.compare(version, "1.73.0") >= 0);
  79. let hasFlexBundleVersion = false;
  80. let flexBundle = {};
  81. if (versionsAllSuitableForFlexBundle) {
  82. hasFlexBundleVersion = true;
  83. const flexBundleResource = await workspace.byPath(`${pathPrefix}/changes/flexibility-bundle.json`);
  84. if (flexBundleResource) {
  85. flexBundle = JSON.parse(await flexBundleResource.getString());
  86. }
  87. }
  88. const processedResources = await flexChangesBundler({
  89. resources: allResources,
  90. options: {
  91. pathPrefix,
  92. hasFlexBundleVersion
  93. },
  94. existingFlexBundle: flexBundle
  95. });
  96. await Promise.all(processedResources.map((resource) => {
  97. log.verbose("Writing flexibility changes bundle");
  98. return workspace.write(resource);
  99. }));
  100. // Add the sap.ui.fl dependency if a bundle has been created
  101. if (processedResources.length > 0) {
  102. await updateFLdependency();
  103. }
  104. // Do not write bundled source files to build result
  105. if (taskUtil) {
  106. allResources.forEach((resource) => {
  107. taskUtil.setTag(resource, taskUtil.STANDARD_TAGS.OmitFromBuildResult);
  108. });
  109. }
  110. }
  111. }