builder/lib/processors/bundlers/moduleBundler.js

  1. import BundleBuilder from "../../lbt/bundle/Builder.js";
  2. import LocatorResourcePool from "../../lbt/resources/LocatorResourcePool.js";
  3. import EvoResource from "@ui5/fs/Resource";
  4. import {getLogger} from "@ui5/logger";
  5. const log = getLogger("builder:processors:bundlers:moduleBundler");
  6. /**
  7. * @public
  8. * @module @ui5/builder/processors/bundlers/moduleBundler
  9. */
  10. /**
  11. * A ModuleBundleDefinitionSection specifies the embedding mode (either 'provided', 'raw', 'preload', 'require'
  12. * or 'bundleInfo') and lists the resources that should be in- or excluded from the section.
  13. * <p>
  14. * <b>Module bundle section modes</b><br>
  15. * <ul>
  16. * <li>
  17. * <code>provided</code>: A section of mode 'provided' defines a set of modules that should not be included in
  18. * the bundle file itself, but which should be assumed to be already loaded (or 'provided') by the environment into
  19. * which the bundle module is loaded.
  20. * </li>
  21. * <li>
  22. * <code>raw</code>: A 'raw' section determines the set of modules that should be embedded, sorts them according
  23. * to their dependencies and writes them out 1:1 without any transformation or wrapping (raw). Only JavaScript
  24. * sources can be embedded in a raw section.
  25. * </li>
  26. * <li>
  27. * <code>preload</code>: A 'preload' section packages resources that should be stored in the preload cache in the
  28. * client. They can embed any textual resource type (JavaScript, XML, JSON and .properties files) that the
  29. * bundling supports. UI5 modules are wrapped into a 'sap.ui.predefine' call. Other JavaScript modules will be
  30. * embedded into a 'jQuery.sap.registerPreload' call, or in a "sap.ui.require.preload" call when
  31. * the ui5loader is available.
  32. * </li>
  33. * <li>
  34. * <code>require</code>: A 'require' section is transformed into a sequence of jQuery.sap.require calls. The
  35. * list will be resolved like an include pattern list in any of the other sections and for each of the resolved
  36. * modules, a jQuery.sap.require will be created. In case the ui5loader is available, 'sap.ui.requireSync' is
  37. * used instead.
  38. * </li>
  39. * <li>
  40. * <code>bundleInfo</code>: A 'bundleInfo' section describes the content of another named bundle. This information
  41. * is transformed into a ui5loader-"bundlesUI5" configuration.
  42. * At runtime, if a module is known to be contained in a bundle, the loader will require that bundle before
  43. * the module itself.
  44. * This requires the ui5loader to be available at build time and UI5 version 1.74.0 or higher at runtime.
  45. * </li>
  46. * </ul>
  47. * </p>
  48. *
  49. * @public
  50. * @typedef {object} ModuleBundleDefinitionSection
  51. * @property {string} mode The embedding mode. Either 'provided', 'raw', 'preload', 'require' or 'bundleInfo'
  52. * @property {string[]} filters List of modules declared as glob patterns (resource name patterns) that should be
  53. * in- or excluded.
  54. * A pattern ending with a slash '/' will, similarly to the use of a single '*' or double '**' asterisk,
  55. * denote an arbitrary number of characters or folder names.
  56. * Excludes should be marked with a leading exclamation mark '!'. The order of filters is relevant; a later
  57. * exclusion overrides an earlier inclusion, and vice versa.
  58. * @example <caption>List of modules as glob patterns that should be in- or excluded</caption>
  59. * // Includes everything from "some/path/to/module/",
  60. * // but excludes the subfolder "some/path/to/module/to/be/excluded/"
  61. * const section = {
  62. * "filters": [
  63. * "some/path/to/module/",
  64. * "!some/path/to/module/to/be/excluded/"
  65. * ]
  66. * };
  67. *
  68. * @property {boolean} [resolve=false] Whether (transitive) dependencies of modules that match the given filters
  69. * should be resolved and added to the module set
  70. * @property {boolean} [resolveConditional=false] Whether conditional dependencies of modules should be resolved
  71. * and added to the module set for this section
  72. * @property {boolean} [renderer=false] Whether renderers for controls should be added to the module set
  73. * @property {boolean} [declareRawModules=false] Whether raw modules should be declared after jQuery.sap.global
  74. * became available. With the usage of the ui5loader, this flag should be set to 'false'
  75. * @property {boolean} [sort=true] Whether the modules should be sorted by their dependencies
  76. */
  77. /* eslint-disable max-len */
  78. /**
  79. * Module bundle definition
  80. *
  81. * @public
  82. * @typedef {object} ModuleBundleDefinition
  83. * @property {string} name The module bundle name
  84. * @property {string[]} [defaultFileTypes=[".js", ".control.xml", ".fragment.html", ".fragment.json", ".fragment.xml", ".view.html", ".view.json", ".view.xml"]]
  85. * List of default file types to be included in the bundle
  86. * @property {module:@ui5/builder/processors/bundlers/moduleBundler~ModuleBundleDefinitionSection[]} sections List of module bundle definition sections.
  87. */
  88. /* eslint-enable max-len */
  89. /**
  90. * Module bundle options
  91. *
  92. * @public
  93. * @typedef {object} ModuleBundleOptions
  94. * @property {boolean} [optimize=true] Whether the module bundle gets minified
  95. * @property {boolean} [sourceMap=true] Whether to generate a source map file for the bundle
  96. * @property {boolean} [decorateBootstrapModule=false] If set to 'false', bootable bundles won't be decorated
  97. * with an optimization marker
  98. * @property {boolean} [addTryCatchRestartWrapper=false] Whether to wrap bootable bundles with
  99. * a try/catch to filter out "Restart" errors
  100. * @property {boolean} [usePredefineCalls=false] If set to 'true', sap.ui.predefine is used for UI5 modules
  101. * @property {number} [numberOfParts=1] The number of parts the module bundle should be splitted
  102. * @property {boolean} [ignoreMissingModules=false] When searching for modules which are optional for further
  103. * processing, do not throw in case they are missing
  104. */
  105. /**
  106. * Result set
  107. *
  108. * @public
  109. * @typedef {object} ModuleBundlerResult
  110. * @property {@ui5/fs/Resource} bundle Bundle resource
  111. * @property {@ui5/fs/Resource} sourceMap Source Map
  112. */
  113. /* eslint-disable max-len */
  114. /**
  115. * Legacy module bundler.
  116. *
  117. * @public
  118. * @function default
  119. * @static
  120. *
  121. * @param {object} parameters Parameters
  122. * @param {@ui5/fs/Resource[]} parameters.resources Resources
  123. * @param {object} parameters.options Options
  124. * @param {object} [parameters.options.moduleNameMapping]
  125. Optional mapping of resource paths to module name in order to overwrite the default determination
  126. * @param {module:@ui5/builder/processors/bundlers/moduleBundler~ModuleBundleDefinition} parameters.options.bundleDefinition Module
  127. bundle definition
  128. * @param {module:@ui5/builder/processors/bundlers/moduleBundler~ModuleBundleOptions} [parameters.options.bundleOptions] Module
  129. bundle options
  130. * @returns {Promise<module:@ui5/builder/processors/bundlers/moduleBundler~ModuleBundlerResult[]>}
  131. * Promise resolving with module bundle resources
  132. */
  133. /* eslint-enable max-len */
  134. export default function({resources, options: {bundleDefinition, bundleOptions, moduleNameMapping}}) {
  135. // Apply defaults without modifying the passed object
  136. bundleOptions = Object.assign({}, {
  137. optimize: true,
  138. sourceMap: true,
  139. decorateBootstrapModule: false,
  140. addTryCatchRestartWrapper: false,
  141. usePredefineCalls: false,
  142. numberOfParts: 1,
  143. ignoreMissingModules: false
  144. }, bundleOptions);
  145. const pool = new LocatorResourcePool({
  146. ignoreMissingModules: bundleOptions.ignoreMissingModules
  147. });
  148. const builder = new BundleBuilder(pool);
  149. if (log.isLevelEnabled("verbose")) {
  150. log.verbose(`Generating bundle:`);
  151. log.verbose(`bundleDefinition: ${JSON.stringify(bundleDefinition, null, 2)}`);
  152. log.verbose(`bundleOptions: ${JSON.stringify(bundleOptions, null, 2)}`);
  153. }
  154. return pool.prepare( resources, moduleNameMapping ).
  155. then( () => builder.createBundle(bundleDefinition, bundleOptions) ).
  156. then( (results) => {
  157. let bundles;
  158. if (results instanceof Array) {
  159. bundles = results;
  160. } else {
  161. bundles = [results];
  162. }
  163. return Promise.all(bundles.map((bundleObj) => {
  164. if ( bundleObj ) {
  165. const {name, content, sourceMap} = bundleObj;
  166. // console.log("creating bundle as '%s'", "/resources/" + name);
  167. const res = {};
  168. res.bundle = new EvoResource({
  169. path: "/resources/" + name,
  170. string: content
  171. });
  172. if (sourceMap) {
  173. res.sourceMap = new EvoResource({
  174. path: "/resources/" + name + ".map",
  175. string: sourceMap
  176. });
  177. }
  178. return res;
  179. }
  180. }));
  181. });
  182. }