Skip to content

Commit

Permalink
feat(core): add dependency builder and external node capability to v2…
Browse files Browse the repository at this point in the history
… api
  • Loading branch information
AgentEnder committed Jul 24, 2023
1 parent ea9f741 commit 619feee
Show file tree
Hide file tree
Showing 14 changed files with 396 additions and 220 deletions.
2 changes: 2 additions & 0 deletions docs/generated/devkit/nx_devkit.md
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,8 @@ Type of dependency between projects

**ProjectGraphBuilder**: `Object`

@deprecated(v18): General project graph processors are deprecated. Replace usage with a plugin that utilizes `processProjectNodes` and `processProjectDependencies`.

---

### Workspaces
Expand Down
2 changes: 2 additions & 0 deletions docs/generated/packages/devkit/documents/nx_devkit.md
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,8 @@ Type of dependency between projects

**ProjectGraphBuilder**: `Object`

@deprecated(v18): General project graph processors are deprecated. Replace usage with a plugin that utilizes `processProjectNodes` and `processProjectDependencies`.

---

### Workspaces
Expand Down
52 changes: 32 additions & 20 deletions packages/nx/src/config/workspaces.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ import {
normalizeProjectRoot,
} from '../project-graph/utils/find-project-for-path';
import { readNxJson } from './nx-json';
import { ProjectGraphExternalNode } from './project-graph';

export class Workspaces {
private cachedProjectsConfig: ProjectsConfigurations;
Expand Down Expand Up @@ -102,7 +103,7 @@ export class Workspaces {
),
this.root,
(path) => readJsonFile(join(this.root, path))
);
).projects;
if (
shouldMergeAngularProjects(
this.root,
Expand Down Expand Up @@ -498,35 +499,41 @@ export function buildProjectsConfigurationsFromProjectPaths(
root: string = workspaceRoot,
readJson: <T extends Object>(string) => T = <T extends Object>(string) =>
readJsonFile<T>(string) // making this an arg allows us to reuse in devkit
): Record<string, ProjectConfiguration> {
): {
projects: Record<string, ProjectConfiguration>;
externalNodes: Record<string, ProjectGraphExternalNode>;
} {
const projectRootMap: Map<string, string> = new Map();
const projects: Record<string, ProjectConfiguration> = {};
const externalNodes: Record<string, ProjectGraphExternalNode> = {};
// We go in reverse here s.t. plugins listed first in the plugins array have highest priority - they overwrite
// whatever configuration was added by plugins later in the array.
const plugins = loadNxPluginsSync(nxJson.plugins).reverse();

// We push the nx core node builder onto the end, s.t. it overwrites any user specified behavior
const globPatternsFromPackageManagerWorkspaces =
getGlobPatternsFromPackageManagerWorkspaces(root);
plugins.push({
const nxCorePlugin: NxPluginV2 = {
name: 'nx-core-build-nodes',
processProjectNodes: {
// Load projects from pnpm / npm workspaces
...(globPatternsFromPackageManagerWorkspaces.length
? ({
? {
[combineGlobPatterns(globPatternsFromPackageManagerWorkspaces)]: (
pkgJsonPath
) => {
const json = readJson<PackageJson>(pkgJsonPath);
return {
[json.name]: buildProjectConfigurationFromPackageJson(
pkgJsonPath,
json,
nxJson
),
projectNodes: {
[json.name]: buildProjectConfigurationFromPackageJson(
pkgJsonPath,
json,
nxJson
),
},
};
},
} as NxPluginV2['processProjectNodes'])
}
: {}),
// Load projects from project.json files. These will be read second, since
// they are listed last in the plugin, so they will overwrite things from the package.json
Expand All @@ -535,36 +542,41 @@ export function buildProjectsConfigurationsFromProjectPaths(
const json = readJson<ProjectConfiguration>(file);
json.name ??= toProjectName(file);
return {
[json.name]: json,
projectNodes: {
[json.name]: json,
},
};
},
},
});
};
plugins.push(nxCorePlugin);

// We iterate over plugins first - this ensures that plugins specified first take precedence.
for (const plugin of plugins) {
// Within a plugin patterns specified later overwrite info from earlier matches.
for (const pattern in plugin.processProjectNodes ?? {}) {
for (const file of projectFiles) {
if (minimatch(file, pattern)) {
const nodes = plugin.processProjectNodes[pattern](file, {
projectsConfigurations: projects,
nxJsonConfiguration: nxJson,
workspaceRoot: root
});
for (const node in nodes) {
const { projectNodes, externalNodes: pluginExternalNodes } =
plugin.processProjectNodes[pattern](file, {
projectsConfigurations: projects,
nxJsonConfiguration: nxJson,
workspaceRoot: root,
});
for (const node in projectNodes) {
mergeProjectConfigurationIntoWorkspace(
projects,
projectRootMap,
nodes[node]
projectNodes[node]
);
}
Object.assign(externalNodes, pluginExternalNodes);
}
}
}
}

return projects;
return { projects, externalNodes };
}

export function mergeTargetConfigurations(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import {
FileData,
ProjectFileMap,
ProjectGraph,
ProjectGraphExternalNode,
} from '../../config/project-graph';
import { buildProjectGraphUsingProjectFileMap } from '../../project-graph/build-project-graph';
import { updateProjectFileMap } from '../../project-graph/file-map-utils';
Expand Down Expand Up @@ -43,6 +44,7 @@ const collectedDeletedFiles = new Set<string>();
let storedWorkspaceConfigHash: string | undefined;
let waitPeriod = 100;
let scheduledTimeoutId;
let knownExternalNodes: Record<string, ProjectGraphExternalNode> = {};

export async function getCachedSerializedProjectGraphPromise() {
try {
Expand Down Expand Up @@ -173,14 +175,12 @@ async function processCollectedUpdatedAndDeletedFiles() {

let nxJson = readNxJson(workspaceRoot);

const projectConfigurations = await retrieveProjectConfigurations(
const { projectNodes } = await retrieveProjectConfigurations(
workspaceRoot,
nxJson
);

const workspaceConfigHash = computeWorkspaceConfigHash(
projectConfigurations
);
const workspaceConfigHash = computeWorkspaceConfigHash(projectNodes);
serverLogger.requestLog(
`Updated file-hasher based on watched changes, recomputing project graph...`
);
Expand All @@ -191,14 +191,12 @@ async function processCollectedUpdatedAndDeletedFiles() {
if (workspaceConfigHash !== storedWorkspaceConfigHash) {
storedWorkspaceConfigHash = workspaceConfigHash;

projectFileMapWithFiles = await retrieveWorkspaceFiles(
workspaceRoot,
nxJson
);
({ externalNodes: knownExternalNodes, ...projectFileMapWithFiles } =
await retrieveWorkspaceFiles(workspaceRoot, nxJson));
} else {
if (projectFileMapWithFiles) {
projectFileMapWithFiles = updateProjectFileMap(
projectConfigurations,
projectNodes,
projectFileMapWithFiles.projectFileMap,
projectFileMapWithFiles.allWorkspaceFiles,
updatedFiles,
Expand Down Expand Up @@ -276,7 +274,8 @@ async function createAndSerializeProjectGraph(): Promise<{
const { projectGraph, projectFileMapCache } =
await buildProjectGraphUsingProjectFileMap(
projectsConfigurations,
projectFileMap,
knownExternalNodes,
projectFileMapWithFiles.projectFileMap,
allWorkspaceFiles,
currentProjectFileMapCache || readProjectFileMapCache(),
true
Expand Down
2 changes: 1 addition & 1 deletion packages/nx/src/generators/utils/project-configuration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -198,7 +198,7 @@ function readAndCombineAllProjectConfigurations(tree: Tree): {
projectFiles,
tree.root,
(file) => readJson(tree, file)
);
).projects;
}

/**
Expand Down
9 changes: 7 additions & 2 deletions packages/nx/src/native/index.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,13 +38,18 @@ export const enum WorkspaceErrors {
Generic = 'Generic'
}
/** Get workspace config files based on provided globs */
export function getProjectConfigurations(workspaceRoot: string, globs: Array<string>, parseConfigurations: (arg0: Array<string>) => Record<string, object>): Record<string, object>
export function getProjectConfigurations(workspaceRoot: string, globs: Array<string>, parseConfigurations: (arg0: Array<string>) => ConfigurationParserResult): ConfigurationParserResult
export interface NxWorkspaceFiles {
projectFileMap: Record<string, Array<FileData>>
globalFiles: Array<FileData>
projectConfigurations: Record<string, object>
externalNodes: Record<string, object>
}
export function getWorkspaceFilesNative(workspaceRoot: string, globs: Array<string>, parseConfigurations: (arg0: Array<string>) => ConfigurationParserResult): NxWorkspaceFiles
export interface ConfigurationParserResult {
projectNodes: Record<string, object>
externalNodes: Record<string, object>
}
export function getWorkspaceFilesNative(workspaceRoot: string, globs: Array<string>, parseConfigurations: (arg0: Array<string>) => Record<string, object>): NxWorkspaceFiles
export class ImportResult {
file: string
sourceProject: string
Expand Down
7 changes: 3 additions & 4 deletions packages/nx/src/native/workspace/get_config_files.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
use crate::native::utils::glob::build_glob_set;
use crate::native::utils::path::Normalize;
use crate::native::walker::nx_walker;
use crate::native::workspace::types::ConfigurationParserResult;
use globset::GlobSet;

use napi::JsObject;
use std::collections::hash_map::Entry;
use std::collections::HashMap;
use std::path::{Path, PathBuf};
Expand All @@ -13,11 +13,10 @@ use std::path::{Path, PathBuf};
pub fn get_project_configurations<ConfigurationParser>(
workspace_root: String,
globs: Vec<String>,

parse_configurations: ConfigurationParser,
) -> napi::Result<HashMap<String, JsObject>>
) -> napi::Result<ConfigurationParserResult>
where
ConfigurationParser: Fn(Vec<String>) -> napi::Result<HashMap<String, JsObject>>,
ConfigurationParser: Fn(Vec<String>) -> napi::Result<ConfigurationParserResult>
{
let globs = build_glob_set(globs)?;
let config_paths: Vec<String> = nx_walker(workspace_root, move |rec| {
Expand Down
21 changes: 12 additions & 9 deletions packages/nx/src/native/workspace/get_nx_workspace_files.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,14 +12,14 @@ use crate::native::utils::glob::build_glob_set;
use crate::native::utils::path::Normalize;
use crate::native::walker::nx_walker;
use crate::native::workspace::errors::WorkspaceErrors;
use crate::native::workspace::get_config_files::insert_config_file_into_map;
use crate::native::workspace::types::FileLocation;
use crate::native::workspace::types::{ConfigurationParserResult, FileLocation};

#[napi(object)]
pub struct NxWorkspaceFiles {
pub project_file_map: HashMap<String, Vec<FileData>>,
pub global_files: Vec<FileData>,
pub project_configurations: HashMap<String, JsObject>,
pub external_nodes: HashMap<String, JsObject>,
}

#[napi]
Expand All @@ -29,7 +29,7 @@ pub fn get_workspace_files_native<ConfigurationParser>(
parse_configurations: ConfigurationParser,
) -> napi::Result<NxWorkspaceFiles, WorkspaceErrors>
where
ConfigurationParser: Fn(Vec<String>) -> napi::Result<HashMap<String, JsObject>>,
ConfigurationParser: Fn(Vec<String>) -> napi::Result<ConfigurationParserResult>,
{
enable_logger();

Expand All @@ -40,10 +40,10 @@ where

let projects_vec: Vec<String> = projects.iter().map(|p| p.to_normalized_string()).collect();

let project_configurations = parse_configurations(projects_vec)
let parsed_graph_nodes = parse_configurations(projects_vec)
.map_err(|e| napi::Error::new(WorkspaceErrors::ParseError, e.to_string()))?;

let root_map = create_root_map(&project_configurations);
let root_map = create_root_map(&parsed_graph_nodes.project_nodes);

trace!(?root_map);

Expand Down Expand Up @@ -94,7 +94,8 @@ where
Ok(NxWorkspaceFiles {
project_file_map,
global_files,
project_configurations,
external_nodes: parsed_graph_nodes.external_nodes,
project_configurations: parsed_graph_nodes.project_nodes,
})
}

Expand All @@ -114,16 +115,18 @@ type WorkspaceData = (HashSet<PathBuf>, Vec<FileData>);
fn get_file_data(workspace_root: &str, globs: Vec<String>) -> anyhow::Result<WorkspaceData> {
let globs = build_glob_set(globs)?;
let (projects, file_data) = nx_walker(workspace_root, move |rec| {
let mut projects: HashMap<PathBuf, PathBuf> = HashMap::new();
let mut projects: HashSet<PathBuf> = HashSet::new();
let mut file_hashes: Vec<FileData> = vec![];
for (path, content) in rec {
file_hashes.push(FileData {
file: path.to_normalized_string(),
hash: xxh3::xxh3_64(&content).to_string(),
});
insert_config_file_into_map(path, &mut projects, &globs)
if globs.is_match(&path) {
projects.insert(path);
}
}
(projects, file_hashes)
});
Ok((projects.into_values().collect(), file_data))
Ok((projects, file_data))
}
10 changes: 10 additions & 0 deletions packages/nx/src/native/workspace/types.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,15 @@
use std::collections::HashMap;

use napi::JsObject;

#[derive(Debug, Eq, PartialEq)]
pub enum FileLocation {
Global,
Project(String),
}

#[napi(object)]
pub struct ConfigurationParserResult {
pub project_nodes: HashMap<String, JsObject>,
pub external_nodes: HashMap<String, JsObject>,
}
7 changes: 7 additions & 0 deletions packages/nx/src/project-graph/build-project-graph.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import { getRootTsConfigPath } from '../plugins/js/utils/typescript';
import {
ProjectFileMap,
ProjectGraph,
ProjectGraphExternalNode,
ProjectGraphProcessorContext,
} from '../config/project-graph';
import { readJsonFile } from '../utils/fileutils';
Expand Down Expand Up @@ -49,6 +50,7 @@ export function getProjectFileMap(): {

export async function buildProjectGraphUsingProjectFileMap(
projectsConfigurations: ProjectsConfigurations,
externalNodes: Record<string, ProjectGraphExternalNode>,
projectFileMap: ProjectFileMap,
allWorkspaceFiles: FileData[],
fileMap: ProjectFileMapCache | null,
Expand Down Expand Up @@ -94,6 +96,7 @@ export async function buildProjectGraphUsingProjectFileMap(
);
let projectGraph = await buildProjectGraphUsingContext(
nxJson,
externalNodes,
context,
cachedFileData,
projectGraphVersion
Expand Down Expand Up @@ -139,6 +142,7 @@ function readCombinedDeps() {

async function buildProjectGraphUsingContext(
nxJson: NxJsonConfiguration,
knownExternalNodes: Record<string, ProjectGraphExternalNode>,
ctx: ProjectGraphProcessorContext,
cachedFileData: { [project: string]: { [file: string]: FileData } },
projectGraphVersion: string
Expand All @@ -147,6 +151,9 @@ async function buildProjectGraphUsingContext(

const builder = new ProjectGraphBuilder(null, ctx.fileMap);
builder.setVersion(projectGraphVersion);
for (const node in knownExternalNodes) {
builder.addExternalNode(knownExternalNodes[node]);
}

await buildWorkspaceProjectNodes(ctx, builder, nxJson);
const initProjectGraph = builder.getUpdatedProjectGraph();
Expand Down
Loading

0 comments on commit 619feee

Please sign in to comment.