UNPKG

nx

Version:

The core Nx plugin contains the core functionality of Nx like the project graph, nx commands and task orchestration.

297 lines (296 loc) 12.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.getPlugins = getPlugins; exports.getPluginsSeparated = getPluginsSeparated; exports.getOnlyDefaultPlugins = getOnlyDefaultPlugins; exports.cleanupPlugins = cleanupPlugins; exports.reasonToError = reasonToError; const node_path_1 = require("node:path"); const angular_json_1 = require("../../adapter/angular-json"); const file_hasher_1 = require("../../hasher/file-hasher"); const workspace_root_1 = require("../../utils/workspace-root"); const in_process_loader_1 = require("./in-process-loader"); const isolation_1 = require("./isolation"); const resolve_plugin_1 = require("./resolve-plugin"); const enabled_1 = require("./isolation/enabled"); const transpiler_1 = require("./transpiler"); /** * Stuff for specified NX Plugins. */ let currentPluginsConfigurationHash; let loadedPlugins; let cachedSeparatedPlugins; let pendingPluginsPromise; let cleanupSpecifiedPlugins; // In-flight separated-plugins load, tagged with its hash. Two roles: a // concurrent caller for the same set shares this load instead of racing a // second one, and it gates the cache commit — a load writes the cache only if // it's still the registered load when it finishes, so a slow older load can't // clobber a newer one's result (two recomputes can overlap). let pendingSeparatedPlugins; const loadingMethod = (plugin, root, index) => (0, enabled_1.isIsolationEnabled)() ? (0, isolation_1.loadIsolatedNxPlugin)(plugin, root, index) : (0, in_process_loader_1.loadNxPlugin)(plugin, root, index); /** * Returns all plugins (specified + default) as a flat list. * Specified plugins come first, followed by default plugins. */ async function getPlugins(nxJson, root = workspace_root_1.workspaceRoot) { const { specifiedPlugins, defaultPlugins } = await getPluginsSeparated(nxJson, root); return specifiedPlugins.concat(defaultPlugins); } /** * Returns specified plugins (from nx.json) and default plugins (project.json, * package.json, etc.) as separate arrays. This separation is needed for * two-phase project configuration processing where target defaults are * applied between specified and default plugin results. * * `nxJson` is required so callers control the snapshot of nx.json the plugin * loader uses. This matters for the daemon's freshness-gated recompute, where * the snap hash and the plugin set must reflect the same disk state. */ async function getPluginsSeparated(nxJson, root = workspace_root_1.workspaceRoot) { const pluginsConfiguration = nxJson.plugins ?? []; const pluginsConfigurationHash = (0, file_hasher_1.hashObject)(pluginsConfiguration); // If the plugins configuration has not changed, reuse the current plugins if (cachedSeparatedPlugins && pluginsConfigurationHash === currentPluginsConfigurationHash) { return cachedSeparatedPlugins; } // A concurrent call is already loading this exact plugin set — share its // load rather than starting a second one that would race the module-level // cache state below. if (pendingSeparatedPlugins?.hash === pluginsConfigurationHash) { return pendingSeparatedPlugins.promise; } // Plugins config changed (e.g. `nx add @nx/maven` updated nx.json). The // cached SeparatedPlugins is invalidated by the early-return above, but // pendingPluginsPromise — the in-flight load — would otherwise be reused // by the `??=` below and serve the previous plugin set forever. Tear // down the old workers and force a fresh load. cleanupSpecifiedPlugins?.(); pendingPluginsPromise = undefined; const loadPromise = (async () => { const results = await Promise.allSettled([ getOnlyDefaultPlugins(root), (pendingPluginsPromise ??= loadSpecifiedNxPlugins(pluginsConfiguration, root)), ]); const errors = []; const defaultPlugins = []; const specifiedPlugins = []; for (let i = 0; i < results.length; i++) { const result = results[i]; if (result.status === 'fulfilled') { (i === 0 ? defaultPlugins : specifiedPlugins).push(...result.value); } else { errors.push(reasonToError(result.reason)); } } if (errors.length > 0) { throw new AggregateError(errors, errors.map((e) => e.message).join('\n')); } const separatedPlugins = { specifiedPlugins, defaultPlugins, }; // Commit only if we're still the registered load — so the hash and the // cached set are always written together and describe the same plugins. if (pendingSeparatedPlugins?.promise === loadPromise) { cachedSeparatedPlugins = separatedPlugins; currentPluginsConfigurationHash = pluginsConfigurationHash; loadedPlugins = specifiedPlugins.concat(defaultPlugins); } return separatedPlugins; })(); pendingSeparatedPlugins = { hash: pluginsConfigurationHash, promise: loadPromise, }; try { return await loadPromise; } finally { // Clear the in-flight marker, but only if it still points at our load — // a newer call may have already replaced it. if (pendingSeparatedPlugins?.promise === loadPromise) { pendingSeparatedPlugins = undefined; } } } /** * Stuff for default NX Plugins. */ let loadedDefaultPlugins; let loadedDefaultPluginsHash; let cleanupDefaultPlugins; let pendingDefaultPluginPromise; async function getOnlyDefaultPlugins(root = workspace_root_1.workspaceRoot) { const hash = root; // If the plugins configuration has not changed, reuse the current plugins if (loadedDefaultPlugins && hash === loadedDefaultPluginsHash) { return loadedDefaultPlugins; } // Cleanup current plugins before loading new ones if (cleanupDefaultPlugins) { cleanupDefaultPlugins(); } pendingDefaultPluginPromise ??= loadDefaultNxPlugins(workspace_root_1.workspaceRoot); const [result, cleanupFn] = await pendingDefaultPluginPromise; cleanupDefaultPlugins = () => { loadedDefaultPlugins = undefined; pendingDefaultPluginPromise = undefined; cleanupFn(); }; loadedDefaultPlugins = result; loadedDefaultPluginsHash = hash; return result; } function cleanupPlugins() { cleanupSpecifiedPlugins?.(); cleanupDefaultPlugins?.(); pendingPluginsPromise = undefined; pendingDefaultPluginPromise = undefined; cachedSeparatedPlugins = undefined; // Drop the in-flight load too: clearing the marker flips its commit gate to // false, so a load resolving after teardown can't repopulate the torn-down cache. pendingSeparatedPlugins = undefined; } /** * Stuff for generic loading */ async function loadDefaultNxPlugins(root = workspace_root_1.workspaceRoot) { performance.mark('loadDefaultNxPlugins:start'); const plugins = getDefaultPlugins(root); const cleanupFunctions = []; const results = await Promise.allSettled(plugins.map(async (plugin) => { performance.mark(`Load Nx Plugin: ${plugin} - start`); const [loadedPluginPromise, cleanup] = await loadingMethod(plugin, root); cleanupFunctions.push(cleanup); const res = await loadedPluginPromise; performance.mark(`Load Nx Plugin: ${plugin} - end`); performance.measure(`Load Nx Plugin: ${plugin}`, `Load Nx Plugin: ${plugin} - start`, `Load Nx Plugin: ${plugin} - end`); return res; })); const defaultPluginResults = []; const errors = []; for (let i = 0; i < results.length; i++) { const result = results[i]; if (result.status === 'fulfilled') { defaultPluginResults.push(result.value); } else { errors.push({ pluginName: plugins[i], error: reasonToError(result.reason), }); } } if (errors.length > 0) { for (const fn of cleanupFunctions) { fn(); } const errorMessage = errors .map((e) => ` - ${e.pluginName}: ${e.error.message}`) .join('\n'); throw new AggregateError(errors.map((e) => e.error), `Failed to load ${errors.length} default Nx plugin(s):\n${errorMessage}`); } const ret = [ defaultPluginResults, () => { for (const fn of cleanupFunctions) { fn(); } if ((0, transpiler_1.pluginTranspilerIsRegistered)()) { (0, transpiler_1.cleanupPluginTSTranspiler)(); } }, ]; performance.mark('loadDefaultNxPlugins:end'); performance.measure('loadDefaultNxPlugins', 'loadDefaultNxPlugins:start', 'loadDefaultNxPlugins:end'); return ret; } async function loadSpecifiedNxPlugins(pluginsConfigurations, root = workspace_root_1.workspaceRoot) { // Returning existing plugins is handled by getPlugins, // so, if we are here and there are existing plugins, they are stale if (cleanupSpecifiedPlugins) { cleanupSpecifiedPlugins(); } performance.mark('loadSpecifiedNxPlugins:start'); pluginsConfigurations ??= []; // Drop the cached workspace-layout snapshot local-plugin resolution uses: // in a long-lived daemon it can predate a newly added local plugin and // resolve it to the workspace root. Runs only when the plugin set changed. (0, resolve_plugin_1.resetResolvePluginCache)(); const cleanupFunctions = []; const results = await Promise.allSettled(pluginsConfigurations.map(async (plugin, index) => { const pluginPath = typeof plugin === 'string' ? plugin : plugin.plugin; performance.mark(`Load Nx Plugin: ${pluginPath} - start`); const [loadedPluginPromise, cleanup] = await loadingMethod(plugin, root, index); cleanupFunctions.push(cleanup); const res = await loadedPluginPromise; performance.mark(`Load Nx Plugin: ${pluginPath} - end`); performance.measure(`Load Nx Plugin: ${pluginPath}`, `Load Nx Plugin: ${pluginPath} - start`, `Load Nx Plugin: ${pluginPath} - end`); return res; })); performance.mark('loadSpecifiedNxPlugins:end'); performance.measure('loadSpecifiedNxPlugins', 'loadSpecifiedNxPlugins:start', 'loadSpecifiedNxPlugins:end'); const plugins = []; const errors = []; for (let i = 0; i < results.length; i++) { const result = results[i]; if (result.status === 'fulfilled') { plugins.push(result.value); } else { const pluginConfig = pluginsConfigurations[i]; const pluginName = typeof pluginConfig === 'string' ? pluginConfig : pluginConfig.plugin; errors.push({ pluginName, error: reasonToError(result.reason), }); } } if (errors.length > 0) { for (const fn of cleanupFunctions) { fn(); } const errorMessage = errors .map((e) => ` - ${e.pluginName}: ${e.error.message}`) .join('\n'); throw new AggregateError(errors.map((e) => e.error), `Failed to load ${errors.length} Nx plugin(s):\n${errorMessage}`); } cleanupSpecifiedPlugins = () => { for (const fn of cleanupFunctions) { fn(); } if ((0, transpiler_1.pluginTranspilerIsRegistered)()) { (0, transpiler_1.cleanupPluginTSTranspiler)(); } pendingPluginsPromise = undefined; }; return plugins; } function reasonToError(reason) { if (reason instanceof Error) { return reason; } if (typeof reason === 'object' && reason !== null && 'message' in reason) { const error = new Error(String(reason.message)); if ('stack' in reason) { error.stack = String(reason.stack); } return error; } return new Error(String(reason)); } function getDefaultPlugins(root) { return [ (0, node_path_1.join)(__dirname, '../../plugins/js'), ...((0, angular_json_1.shouldMergeAngularProjects)(root, false) ? [(0, node_path_1.join)(__dirname, '../../adapter/angular-json')] : []), (0, node_path_1.join)(__dirname, '../../plugins/package-json'), (0, node_path_1.join)(__dirname, '../../plugins/project-json/build-nodes/project-json'), ]; }