project/lib/graph/graph.js

  1. import path from "node:path";
  2. import projectGraphBuilder from "./projectGraphBuilder.js";
  3. import ui5Framework from "./helpers/ui5Framework.js";
  4. import createWorkspace from "./helpers/createWorkspace.js";
  5. import {getLogger} from "@ui5/logger";
  6. const log = getLogger("generateProjectGraph");
  7. /**
  8. * Helper module to create a [@ui5/project/graph/ProjectGraph]{@link @ui5/project/graph/ProjectGraph}
  9. * from a directory
  10. *
  11. * @public
  12. * @module @ui5/project/graph
  13. */
  14. /**
  15. * Generates a [@ui5/project/graph/ProjectGraph]{@link @ui5/project/graph/ProjectGraph} by resolving
  16. * dependencies from package.json files and configuring projects from ui5.yaml files
  17. *
  18. * @public
  19. * @static
  20. * @param {object} [options]
  21. * @param {string} [options.cwd=process.cwd()] Directory to start searching for the root module
  22. * @param {object} [options.rootConfiguration]
  23. * Configuration object to use for the root module instead of reading from a configuration file
  24. * @param {string} [options.rootConfigPath]
  25. * Configuration file to use for the root module instead the default ui5.yaml. Either a path relative to
  26. * <code>cwd</code> or an absolute path. In both case, platform-specific path segment separators must be used.
  27. * @param {string} [options.versionOverride] Framework version to use instead of the one defined in the root project
  28. * @param {string} [options.resolveFrameworkDependencies=true]
  29. * Whether framework dependencies should be added to the graph
  30. * @param {string} [options.workspaceName]
  31. * Name of the workspace configuration that should be used if any is provided
  32. * @param {module:@ui5/project/ui5Framework/maven/CacheMode} [options.cacheMode]
  33. * Cache mode to use when consuming SNAPSHOT versions of a framework
  34. * @param {string} [options.workspaceConfigPath=ui5-workspace.yaml]
  35. * Workspace configuration file to use if no object has been provided
  36. * @param {@ui5/project/graph/Workspace~Configuration} [options.workspaceConfiguration]
  37. * Workspace configuration object to use instead of reading from a configuration file.
  38. * Parameter <code>workspaceName</code> can either be omitted or has to match with the given configuration name
  39. * @returns {Promise<@ui5/project/graph/ProjectGraph>} Promise resolving to a Project Graph instance
  40. */
  41. export async function graphFromPackageDependencies({
  42. cwd, rootConfiguration, rootConfigPath,
  43. versionOverride, cacheMode, resolveFrameworkDependencies = true,
  44. workspaceName /* TODO 4.0: default workspaceName to "default" ? */,
  45. workspaceConfiguration, workspaceConfigPath = "ui5-workspace.yaml"
  46. }) {
  47. log.verbose(`Creating project graph using npm provider...`);
  48. const {
  49. default: NpmProvider
  50. } = await import("./providers/NodePackageDependencies.js");
  51. cwd = cwd ? path.resolve(cwd) : process.cwd();
  52. rootConfigPath = utils.resolveConfigPath(cwd, rootConfigPath);
  53. let workspace;
  54. if (workspaceName || workspaceConfiguration) {
  55. workspace = await createWorkspace({
  56. cwd,
  57. name: workspaceName,
  58. configObject: workspaceConfiguration,
  59. configPath: workspaceConfigPath
  60. });
  61. }
  62. const provider = new NpmProvider({
  63. cwd,
  64. rootConfiguration,
  65. rootConfigPath
  66. });
  67. const projectGraph = await projectGraphBuilder(provider, workspace);
  68. if (resolveFrameworkDependencies) {
  69. await ui5Framework.enrichProjectGraph(projectGraph, {versionOverride, cacheMode, workspace});
  70. }
  71. return projectGraph;
  72. }
  73. /**
  74. * Generates a [@ui5/project/graph/ProjectGraph]{@link @ui5/project/graph/ProjectGraph} from a
  75. * YAML file following the structure of
  76. * [@ui5/project/graph/providers/DependencyTree~TreeNode]{@link @ui5/project/graph/providers/DependencyTree~TreeNode}.
  77. *
  78. * Documentation:
  79. * [Static Dependency Definition]{@link https://sap.github.io/ui5-tooling/stable/pages/Overview/#static-dependency-definition}
  80. *
  81. * @public
  82. * @static
  83. * @param {object} options
  84. * @param {object} [options.filePath=projectDependencies.yaml] Path to the dependency configuration file
  85. * @param {string} [options.cwd=process.cwd()] Directory to resolve relative paths to
  86. * @param {object} [options.rootConfiguration]
  87. * Configuration object to use for the root module instead of reading from a configuration file
  88. * @param {string} [options.rootConfigPath]
  89. * Configuration file to use for the root module instead the default ui5.yaml. Either a path relative to
  90. * <code>cwd</code> or an absolute path. In both case, platform-specific path segment separators must be used.
  91. * @param {string} [options.versionOverride] Framework version to use instead of the one defined in the root project
  92. * @param {module:@ui5/project/ui5Framework/maven/CacheMode} [options.cacheMode]
  93. * Cache mode to use when consuming SNAPSHOT versions of a framework
  94. * @param {string} [options.resolveFrameworkDependencies=true]
  95. * Whether framework dependencies should be added to the graph
  96. * @returns {Promise<@ui5/project/graph/ProjectGraph>} Promise resolving to a Project Graph instance
  97. */
  98. export async function graphFromStaticFile({
  99. filePath = "projectDependencies.yaml", cwd,
  100. rootConfiguration, rootConfigPath,
  101. versionOverride, cacheMode, resolveFrameworkDependencies = true
  102. }) {
  103. log.verbose(`Creating project graph using static file...`);
  104. const {
  105. default: DependencyTreeProvider
  106. } = await import("./providers/DependencyTree.js");
  107. cwd = cwd ? path.resolve(cwd) : process.cwd();
  108. rootConfigPath = utils.resolveConfigPath(cwd, rootConfigPath);
  109. const dependencyTree = await utils.readDependencyConfigFile(cwd, filePath);
  110. const provider = new DependencyTreeProvider({
  111. dependencyTree,
  112. rootConfiguration,
  113. rootConfigPath
  114. });
  115. const projectGraph = await projectGraphBuilder(provider);
  116. if (resolveFrameworkDependencies) {
  117. await ui5Framework.enrichProjectGraph(projectGraph, {versionOverride, cacheMode});
  118. }
  119. return projectGraph;
  120. }
  121. /**
  122. * Generates a [@ui5/project/graph/ProjectGraph]{@link @ui5/project/graph/ProjectGraph} from the
  123. * given <code>dependencyTree</code> following the structure of
  124. * [@ui5/project/graph/providers/DependencyTree~TreeNode]{@link @ui5/project/graph/providers/DependencyTree~TreeNode}
  125. *
  126. * @public
  127. * @static
  128. * @param {object} options
  129. * @param {@ui5/project/graph/providers/DependencyTree~TreeNode} options.dependencyTree
  130. * @param {string} [options.cwd=process.cwd()] Directory to resolve relative paths to
  131. * @param {object} [options.rootConfiguration]
  132. * Configuration object to use for the root module instead of reading from a configuration file
  133. * @param {string} [options.rootConfigPath]
  134. * Configuration file to use for the root module instead the default ui5.yaml. Either a path relative to
  135. * <code>cwd</code> or an absolute path. In both case, platform-specific path segment separators must be used.
  136. * @param {string} [options.versionOverride] Framework version to use instead of the one defined in the root project
  137. * @param {module:@ui5/project/ui5Framework/maven/CacheMode} [options.cacheMode]
  138. * Cache mode to use when consuming SNAPSHOT versions of a framework
  139. * @param {string} [options.resolveFrameworkDependencies=true]
  140. * Whether framework dependencies should be added to the graph
  141. * @returns {Promise<@ui5/project/graph/ProjectGraph>} Promise resolving to a Project Graph instance
  142. */
  143. export async function graphFromObject({
  144. dependencyTree, cwd,
  145. rootConfiguration, rootConfigPath,
  146. versionOverride, cacheMode, resolveFrameworkDependencies = true
  147. }) {
  148. log.verbose(`Creating project graph using object...`);
  149. const {
  150. default: DependencyTreeProvider
  151. } = await import("./providers/DependencyTree.js");
  152. cwd = cwd ? path.resolve(cwd) : process.cwd();
  153. rootConfigPath = utils.resolveConfigPath(cwd, rootConfigPath);
  154. const dependencyTreeProvider = new DependencyTreeProvider({
  155. dependencyTree,
  156. rootConfiguration,
  157. rootConfigPath
  158. });
  159. const projectGraph = await projectGraphBuilder(dependencyTreeProvider);
  160. if (resolveFrameworkDependencies) {
  161. await ui5Framework.enrichProjectGraph(projectGraph, {versionOverride, cacheMode});
  162. }
  163. return projectGraph;
  164. }
  165. const utils = {
  166. resolveConfigPath: function(cwd, configPath) {
  167. if (configPath && !path.isAbsolute(configPath)) {
  168. configPath = path.join(cwd, configPath);
  169. }
  170. return configPath;
  171. },
  172. readDependencyConfigFile: async function(cwd, filePath) {
  173. const {
  174. default: fs
  175. } = await import("graceful-fs");
  176. const {promisify} = await import("util");
  177. const readFile = promisify(fs.readFile);
  178. const parseYaml =(await import("js-yaml")).load;
  179. filePath = utils.resolveConfigPath(cwd, filePath);
  180. let dependencyTree;
  181. try {
  182. const contents = await readFile(filePath, {encoding: "utf-8"});
  183. dependencyTree = parseYaml(contents, {
  184. filename: filePath
  185. });
  186. utils.resolveProjectPaths(cwd, dependencyTree);
  187. } catch (err) {
  188. throw new Error(
  189. `Failed to load dependency tree configuration from path ${filePath}: ${err.message}`);
  190. }
  191. return dependencyTree;
  192. },
  193. resolveProjectPaths: function(cwd, project) {
  194. if (!project.path) {
  195. throw new Error(`Missing or empty attribute 'path' for project ${project.id}`);
  196. }
  197. project.path = path.resolve(cwd, project.path);
  198. if (!project.id) {
  199. throw new Error(`Missing or empty attribute 'id' for project with path ${project.path}`);
  200. }
  201. if (!project.version) {
  202. throw new Error(`Missing or empty attribute 'version' for project ${project.id}`);
  203. }
  204. if (project.dependencies) {
  205. project.dependencies.forEach((project) => utils.resolveProjectPaths(cwd, project));
  206. }
  207. return project;
  208. }
  209. };
  210. // Export function for testing only
  211. /* istanbul ignore else */
  212. if (process.env.NODE_ENV === "test") {
  213. graphFromStaticFile._utils = utils;
  214. }