project/lib/graph/projectGraphBuilder.js

  1. import path from "node:path";
  2. import Module from "./Module.js";
  3. import ProjectGraph from "./ProjectGraph.js";
  4. import ShimCollection from "./ShimCollection.js";
  5. import {getLogger} from "@ui5/logger";
  6. const log = getLogger("graph:projectGraphBuilder");
  7. function _handleExtensions(graph, shimCollection, extensions) {
  8. extensions.forEach((extension) => {
  9. const type = extension.getType();
  10. switch (type) {
  11. case "project-shim":
  12. shimCollection.addProjectShim(extension);
  13. break;
  14. case "task":
  15. case "server-middleware":
  16. graph.addExtension(extension);
  17. break;
  18. default:
  19. throw new Error(
  20. `Encountered unexpected extension of type ${type} ` +
  21. `Supported types are 'project-shim', 'task' and 'middleware'`);
  22. }
  23. });
  24. }
  25. function validateNode(node) {
  26. if (node.specVersion) {
  27. throw new Error(
  28. `Provided node with ID ${node.id} contains a top-level 'specVersion' property. ` +
  29. `With UI5 Tooling 3.0, project configuration needs to be provided in a dedicated ` +
  30. `'configuration' object`);
  31. }
  32. if (node.metadata) {
  33. throw new Error(
  34. `Provided node with ID ${node.id} contains a top-level 'metadata' property. ` +
  35. `With UI5 Tooling 3.0, project configuration needs to be provided in a dedicated ` +
  36. `'configuration' object`);
  37. }
  38. }
  39. /**
  40. * @public
  41. * @module @ui5/project/graph/ProjectGraphBuilder
  42. */
  43. /**
  44. * Dependency graph node representing a module
  45. *
  46. * @public
  47. * @typedef {object} @ui5/project/graph/ProjectGraphBuilder~Node
  48. * @property {string} node.id Unique ID for the project
  49. * @property {string} node.version Version of the project
  50. * @property {string} node.path File System path to access the projects resources
  51. * @property {object|object[]} [node.configuration]
  52. * Configuration object or array of objects to use instead of reading from a configuration file
  53. * @property {string} [node.configPath] Configuration file to use instead the default ui5.yaml
  54. * @property {boolean} [node.optional]
  55. * Whether the node is an optional dependency of the parent it has been requested for
  56. * @property {*} * Additional attributes are allowed but ignored.
  57. * These can be used to pass information internally in the provider.
  58. */
  59. /**
  60. * Node Provider interface
  61. *
  62. * @public
  63. * @interface @ui5/project/graph/ProjectGraphBuilder~NodeProvider
  64. */
  65. /**
  66. * Retrieve information on the root module
  67. *
  68. * @public
  69. * @function
  70. * @name @ui5/project/graph/ProjectGraphBuilder~NodeProvider#getRootNode
  71. * @returns {Node} The root node of the dependency graph
  72. */
  73. /**
  74. * Retrieve information on given a nodes dependencies
  75. *
  76. * @public
  77. * @function
  78. * @name @ui5/project/graph/ProjectGraphBuilder~NodeProvider#getDependencies
  79. * @param {Node} node The root node of the dependency graph
  80. * @param {@ui5/project/graph/Workspace} [workspace] workspace instance to use for overriding node resolution
  81. * @returns {Node[]} Array of nodes which are direct dependencies of the given node
  82. */
  83. /**
  84. * Generic helper module to create a [@ui5/project/graph/ProjectGraph]{@link @ui5/project/graph/ProjectGraph}.
  85. * For example from a dependency tree as returned by the legacy "translators".
  86. *
  87. * @public
  88. * @function default
  89. * @static
  90. * @param {@ui5/project/graph/ProjectGraphBuilder~NodeProvider} nodeProvider
  91. * Node provider instance to use for building the graph
  92. * @param {@ui5/project/graph/Workspace} [workspace]
  93. * Optional workspace instance to use for overriding project resolutions
  94. * @returns {@ui5/project/graph/ProjectGraph} A new project graph instance
  95. */
  96. async function projectGraphBuilder(nodeProvider, workspace) {
  97. const shimCollection = new ShimCollection();
  98. const moduleCollection = Object.create(null);
  99. const handledExtensions = new Set(); // Set containing the IDs of modules which' extensions have been handled
  100. const rootNode = await nodeProvider.getRootNode();
  101. validateNode(rootNode);
  102. const rootModule = new Module({
  103. id: rootNode.id,
  104. version: rootNode.version,
  105. modulePath: rootNode.path,
  106. configPath: rootNode.configPath,
  107. configuration: rootNode.configuration
  108. });
  109. const {project: rootProject, extensions: rootExtensions} = await rootModule.getSpecifications();
  110. if (!rootProject) {
  111. throw new Error(
  112. `Failed to create a UI5 project from module ${rootNode.id} at ${rootNode.path}. ` +
  113. `Make sure the path is correct and a project configuration is present or supplied.`);
  114. }
  115. moduleCollection[rootNode.id] = rootModule;
  116. const rootProjectName = rootProject.getName();
  117. let qualifiedApplicationProject = null;
  118. if (rootProject.getType() === "application") {
  119. log.verbose(`Root project ${rootProjectName} qualified as application project for project graph`);
  120. qualifiedApplicationProject = rootProject;
  121. }
  122. const projectGraph = new ProjectGraph({
  123. rootProjectName: rootProjectName
  124. });
  125. projectGraph.addProject(rootProject);
  126. function handleExtensions(extensions) {
  127. return _handleExtensions(projectGraph, shimCollection, extensions);
  128. }
  129. handleExtensions(rootExtensions);
  130. handledExtensions.add(rootNode.id);
  131. const queue = [];
  132. const rootDependencies = await nodeProvider.getDependencies(rootNode, workspace);
  133. if (rootDependencies && rootDependencies.length) {
  134. queue.push({
  135. nodes: rootDependencies,
  136. parentProject: rootProject
  137. });
  138. }
  139. // Breadth-first search
  140. while (queue.length) {
  141. const {nodes, parentProject} = queue.shift(); // Get and remove first entry from queue
  142. const res = await Promise.all(nodes.map(async (node) => {
  143. let ui5Module = moduleCollection[node.id];
  144. if (ui5Module) {
  145. log.silly(
  146. `Re-visiting module ${node.id} as a dependency of ${parentProject.getName()}`);
  147. const {project, extensions} = await ui5Module.getSpecifications();
  148. if (!project && !extensions.length) {
  149. // Invalidate cache if the cached module is visited through another parent project and did not
  150. // resolve to a project or extension(s) before.
  151. // The module being visited now might be a different version containing for example
  152. // UI5 Tooling configuration, or one of the parent projects could have defined a
  153. // relevant configuration shim meanwhile
  154. log.silly(
  155. `Cached module ${node.id} did not resolve to any projects or extensions. ` +
  156. `Recreating module as a dependency of ${parentProject.getName()}...`);
  157. ui5Module = null;
  158. }
  159. }
  160. if (!ui5Module) {
  161. log.silly(`Visiting Module ${node.id} as a dependency of ${parentProject.getName()}`);
  162. log.verbose(`Creating module ${node.id}...`);
  163. validateNode(node);
  164. ui5Module = moduleCollection[node.id] = new Module({
  165. id: node.id,
  166. version: node.version,
  167. modulePath: node.path,
  168. configPath: node.configPath,
  169. configuration: node.configuration,
  170. shimCollection
  171. });
  172. } else if (ui5Module.getPath() !== node.path) {
  173. log.verbose(
  174. `Warning - Dependency ${node.id} is available at multiple paths:` +
  175. `\n Location of the already processed module (this one will be used): ${ui5Module.getPath()}` +
  176. `\n Additional location (this one will be ignored): ${node.path}`);
  177. }
  178. const {project, extensions} = await ui5Module.getSpecifications();
  179. return {
  180. node,
  181. project,
  182. extensions
  183. };
  184. }));
  185. // Keep this out of the async map function to ensure
  186. // all projects and extensions are applied in a deterministic order
  187. for (let i = 0; i < res.length; i++) {
  188. const {
  189. node, // Tree "raw" dependency tree node
  190. project, // The project found for this node, if any
  191. extensions // Any extensions found for this node
  192. } = res[i];
  193. if (extensions.length && (!node.optional || parentProject === rootProject)) {
  194. // Only handle extensions in non-optional dependencies and any dependencies of the root project
  195. if (handledExtensions.has(node.id)) {
  196. // Do not handle extensions of the same module twice
  197. log.verbose(`Extensions contained in module ${node.id} have already been handled`);
  198. } else {
  199. log.verbose(`Handling extensions for module ${node.id}...`);
  200. // If a different module contains the same extension, we expect an error to be thrown by the graph
  201. handleExtensions(extensions);
  202. handledExtensions.add(node.id);
  203. }
  204. }
  205. // Check for collection shims
  206. const collectionShims = shimCollection.getCollectionShims(node.id);
  207. if (collectionShims && collectionShims.length) {
  208. log.verbose(
  209. `One or more module collection shims have been defined for module ${node.id}. ` +
  210. `Therefore the module itself will not be resolved.`);
  211. const shimmedNodes = collectionShims.map(({name, shim}) => {
  212. log.verbose(`Applying module collection shim ${name} for module ${node.id}:`);
  213. return Object.entries(shim.modules).map(([shimModuleId, shimModuleRelPath]) => {
  214. const shimModulePath = path.join(node.path, shimModuleRelPath);
  215. log.verbose(` Injecting module ${shimModuleId} with path ${shimModulePath}`);
  216. return {
  217. id: shimModuleId,
  218. version: node.version,
  219. path: shimModulePath
  220. };
  221. });
  222. });
  223. queue.push({
  224. nodes: Array.prototype.concat.apply([], shimmedNodes),
  225. parentProject,
  226. });
  227. // Skip collection node
  228. continue;
  229. }
  230. let skipDependencies = false;
  231. if (project) {
  232. const projectName = project.getName();
  233. if (project.getType() === "application") {
  234. // Special handling of application projects of which there must be exactly *one*
  235. // in the graph. Others shall be ignored.
  236. if (!qualifiedApplicationProject) {
  237. log.verbose(`Project ${projectName} qualified as application project for project graph`);
  238. qualifiedApplicationProject = project;
  239. } else if (qualifiedApplicationProject.getName() !== projectName) {
  240. // Project is not a duplicate of an already qualified project (which should
  241. // still be processed below), but a unique, additional application project
  242. // TODO: Should this rather be a verbose logging?
  243. // projectPreprocessor handled this like any project that got ignored and did a
  244. // (in this case misleading) general verbose logging:
  245. // "Ignoring project with missing configuration"
  246. log.info(
  247. `Excluding additional application project ${projectName} from graph. `+
  248. `The project graph can only feature a single project of type application. ` +
  249. `Project ${qualifiedApplicationProject.getName()} has already qualified for that role.`);
  250. continue;
  251. }
  252. }
  253. if (projectGraph.getProject(projectName)) {
  254. // Opposing to extensions, we are generally fine with the same project being contained in different
  255. // modules. We simply ignore all but the first occurrence.
  256. // This can happen for example if the same project is packaged in different ways/modules
  257. // (e.g. one module containing the source and one containing the pre-built resources)
  258. log.verbose(
  259. `Project ${projectName} has already been added to the graph. ` +
  260. `Skipping dependency resolution...`);
  261. skipDependencies = true;
  262. } else {
  263. projectGraph.addProject(project);
  264. }
  265. if (parentProject) {
  266. if (node.optional) {
  267. projectGraph.declareOptionalDependency(parentProject.getName(), projectName);
  268. } else {
  269. projectGraph.declareDependency(parentProject.getName(), projectName);
  270. }
  271. if (project.isDeprecated() && parentProject === rootProject &&
  272. parentProject.getName() !== "testsuite") {
  273. // Only warn for direct dependencies of the root project
  274. // No warning for testsuite projects
  275. log.warn(
  276. `Dependency ${project.getName()} is deprecated and should not be used for new projects!`);
  277. }
  278. if (project.isSapInternal() && parentProject === rootProject &&
  279. !parentProject.getAllowSapInternal()) {
  280. // Only warn for direct dependencies of the root project, except it defines "allowSapInternal"
  281. log.warn(
  282. `Dependency ${project.getName()} is restricted for use by SAP internal projects only! ` +
  283. `If the project ${parentProject.getName()} is an SAP internal project, add the attribute ` +
  284. `"allowSapInternal: true" to its metadata configuration`);
  285. }
  286. }
  287. }
  288. if (!project && !extensions.length) {
  289. // Module provides neither a project nor an extension
  290. // => Do not follow its dependencies
  291. log.verbose(
  292. `Module ${node.id} neither provides a project nor an extension. Skipping dependency resolution`);
  293. skipDependencies = true;
  294. }
  295. if (skipDependencies) {
  296. continue;
  297. }
  298. const nodeDependencies = await nodeProvider.getDependencies(node);
  299. if (nodeDependencies && nodeDependencies.length) {
  300. queue.push({
  301. // copy array, so that the queue is stable while ignored project dependencies are removed
  302. nodes: [...nodeDependencies],
  303. parentProject: project ? project : parentProject,
  304. });
  305. }
  306. }
  307. }
  308. // Apply dependency shims
  309. for (const [shimmedModuleId, moduleDepShims] of Object.entries(shimCollection.getAllDependencyShims())) {
  310. const sourceModule = moduleCollection[shimmedModuleId];
  311. for (let j = 0; j < moduleDepShims.length; j++) {
  312. const depShim = moduleDepShims[j];
  313. if (!sourceModule) {
  314. log.warn(`Could not apply dependency shim ${depShim.name} for ${shimmedModuleId}: ` +
  315. `Module ${shimmedModuleId} is unknown`);
  316. continue;
  317. }
  318. const {project: sourceProject} = await sourceModule.getSpecifications();
  319. if (!sourceProject) {
  320. log.warn(`Could not apply dependency shim ${depShim.name} for ${shimmedModuleId}: ` +
  321. `Source module ${shimmedModuleId} does not contain a project`);
  322. continue;
  323. }
  324. for (let k = 0; k < depShim.shim.length; k++) {
  325. const targetModuleId = depShim.shim[k];
  326. const targetModule = moduleCollection[targetModuleId];
  327. if (!targetModule) {
  328. log.warn(`Could not apply dependency shim ${depShim.name} for ${shimmedModuleId}: ` +
  329. `Target module $${depShim} is unknown`);
  330. continue;
  331. }
  332. const {project: targetProject} = await targetModule.getSpecifications();
  333. if (!targetProject) {
  334. log.warn(`Could not apply dependency shim ${depShim.name} for ${shimmedModuleId}: ` +
  335. `Target module ${targetModuleId} does not contain a project`);
  336. continue;
  337. }
  338. projectGraph.declareDependency(sourceProject.getName(), targetProject.getName());
  339. }
  340. }
  341. }
  342. await projectGraph.resolveOptionalDependencies();
  343. return projectGraph;
  344. }
  345. export default projectGraphBuilder;