builder/lib/tasks/jsdoc/generateJsdoc.js

  1. import {getLogger} from "@ui5/logger";
  2. const log = getLogger("builder:tasks:jsdoc:generateJsdoc");
  3. import path from "node:path";
  4. import os from "node:os";
  5. import fs from "graceful-fs";
  6. import {rimraf} from "rimraf";
  7. import {promisify} from "node:util";
  8. const mkdtemp = promisify(fs.mkdtemp);
  9. const mkdir = promisify(fs.mkdir);
  10. import jsdocGenerator from "../../processors/jsdoc/jsdocGenerator.js";
  11. import {createAdapter} from "@ui5/fs/resourceFactory";
  12. /**
  13. * @public
  14. * @module @ui5/builder/tasks/jsdoc/generateJsdoc
  15. */
  16. /**
  17. *
  18. * @public
  19. * @typedef {object} GenerateJsdocOptions
  20. * @property {string|string[]} pattern Pattern to locate the files to be processed
  21. * @property {string} projectName Project name
  22. * @property {string} namespace Namespace to build (e.g. <code>some/project/name</code>)
  23. * @property {string} version Project version
  24. */
  25. /**
  26. * Task to execute a JSDoc build for UI5 projects
  27. *
  28. * @public
  29. * @function default
  30. * @static
  31. *
  32. * @param {object} parameters Parameters
  33. * @param {@ui5/fs/DuplexCollection} parameters.workspace DuplexCollection to read and write files
  34. * @param {@ui5/fs/AbstractReader} parameters.dependencies Reader or Collection to read dependency files
  35. * @param {module:@ui5/builder/tasks/jsdoc/generateJsdoc~GenerateJsdocOptions} parameters.options Options
  36. * @param {@ui5/project/build/helpers/TaskUtil|object} [parameters.taskUtil] TaskUtil
  37. * @returns {Promise<undefined>} Promise resolving with <code>undefined</code> once data has been written
  38. */
  39. export default async function generateJsdoc({
  40. taskUtil,
  41. workspace,
  42. dependencies,
  43. options = {}
  44. }) {
  45. const {projectName, namespace, version, pattern} =
  46. /** @type {module:@ui5/builder/tasks/jsdoc/generateJsdoc~GenerateJsdocOptions} */ (options);
  47. if (!projectName || !namespace || !version || !pattern) {
  48. throw new Error("[generateJsdoc]: One or more mandatory options not provided");
  49. }
  50. const {sourcePath: resourcePath, targetPath, tmpPath, cleanup} =
  51. await utils.createTmpDirs(projectName);
  52. taskUtil?.registerCleanupTask(cleanup);
  53. const [writtenResourcesCount] = await Promise.all([
  54. utils.writeResourcesToDir({
  55. workspace,
  56. pattern,
  57. targetPath: resourcePath
  58. }),
  59. utils.writeDependencyApisToDir({
  60. dependencies,
  61. targetPath: path.join(tmpPath, "dependency-apis")
  62. })
  63. ]);
  64. if (writtenResourcesCount === 0) {
  65. log.info(`Failed to find any input resources for project ${projectName} using pattern ` +
  66. `${pattern}. Skipping JSDoc generation...`);
  67. return;
  68. }
  69. const createdResources = await jsdocGenerator({
  70. sourcePath: resourcePath,
  71. targetPath,
  72. tmpPath,
  73. options: {
  74. projectName,
  75. namespace,
  76. version,
  77. variants: ["apijson"]
  78. }
  79. });
  80. await Promise.all(createdResources.map((resource) => {
  81. return workspace.write(resource);
  82. }));
  83. }
  84. const utils = {
  85. /**
  86. * Create temporary directories for JSDoc generation processor
  87. *
  88. * @private
  89. * @param {string} projectName Project name used for naming the temporary working directory
  90. * @returns {Promise<object>} Promise resolving with sourcePath, targetPath and tmpPath strings
  91. */
  92. createTmpDirs: async function(projectName) {
  93. const tmpDirPath = await utils.createTmpDir(projectName);
  94. const sourcePath = path.join(tmpDirPath, "src"); // dir will be created by writing project resources below
  95. await mkdir(sourcePath, {recursive: true});
  96. const targetPath = path.join(tmpDirPath, "target"); // dir will be created by jsdoc itself
  97. await mkdir(targetPath, {recursive: true});
  98. const tmpPath = path.join(tmpDirPath, "tmp"); // dir needs to be created by us
  99. await mkdir(tmpPath, {recursive: true});
  100. return {
  101. sourcePath,
  102. targetPath,
  103. tmpPath,
  104. cleanup: async () => {
  105. return rimraf(tmpDirPath);
  106. }
  107. };
  108. },
  109. /**
  110. * Create a temporary directory on the host system
  111. *
  112. * @private
  113. * @param {string} projectName Project name used for naming the temporary directory
  114. * @returns {Promise<string>} Promise resolving with path of the temporary directory
  115. */
  116. createTmpDir: async function(projectName) {
  117. const sanitizedProjectName = projectName.replace(/[^A-Za-z0-9]/g, "");
  118. const tmpRootPath = path.join(os.tmpdir(), "ui5-tooling");
  119. await mkdir(tmpRootPath, {recursive: true});
  120. // Appending minus sign also because node docs advise to "avoid trailing X characters in prefix"
  121. return mkdtemp(path.join(tmpRootPath, `jsdoc-${sanitizedProjectName}-`));
  122. },
  123. /**
  124. * Write resources from workspace matching the given pattern to the given fs destination
  125. *
  126. * @private
  127. * @param {object} parameters Parameters
  128. * @param {@ui5/fs/DuplexCollection} parameters.workspace DuplexCollection to read and write files
  129. * @param {string} parameters.pattern Pattern to match resources in workspace against
  130. * @param {string} parameters.targetPath Path to write the resources to
  131. * @returns {Promise<number>} Promise resolving with number of resources written to given directory
  132. */
  133. writeResourcesToDir: async function({workspace, pattern, targetPath}) {
  134. const fsTarget = createAdapter({
  135. fsBasePath: targetPath,
  136. virBasePath: "/resources/"
  137. });
  138. const allResources = await workspace.byGlob(pattern);
  139. // write all resources to the tmp folder
  140. await Promise.all(allResources.map((resource) => fsTarget.write(resource)));
  141. return allResources.length;
  142. },
  143. /**
  144. * Write api.json files of dependencies to given target path in a flat structure
  145. *
  146. * @private
  147. * @param {object} parameters Parameters
  148. * @param {@ui5/fs/AbstractReader} parameters.dependencies Reader or Collection to read dependency files
  149. * @param {string} parameters.targetPath Path to write the resources to
  150. * @returns {Promise<number>} Promise resolving with number of resources written to given directory
  151. */
  152. writeDependencyApisToDir: async function({dependencies, targetPath}) {
  153. const depApis = await dependencies.byGlob("/test-resources/**/designtime/api.json");
  154. // Clone resources before changing their path
  155. const apis = await Promise.all(depApis.map((resource) => resource.clone()));
  156. for (let i = 0; i < apis.length; i++) {
  157. apis[i].setPath(`/api-${i}.json`);
  158. }
  159. const fsTarget = createAdapter({
  160. fsBasePath: targetPath,
  161. virBasePath: "/"
  162. });
  163. await Promise.all(apis.map((resource) => fsTarget.write(resource)));
  164. return apis.length;
  165. }
  166. };
  167. // Export utils for testing only
  168. /* istanbul ignore else */
  169. if (process.env.NODE_ENV === "test") {
  170. generateJsdoc._utils = utils;
  171. }