From 04cf256c31aaa012788d8abb198f744b38e56b51 Mon Sep 17 00:00:00 2001 From: Craigory Coppola Date: Thu, 25 Jul 2024 18:35:03 -0400 Subject: [PATCH] feat(devkit): allow updating package json based projects --- .nxignore | 2 +- .../utils/project-configuration.spec.ts | 52 ++++++++++ .../generators/utils/project-configuration.ts | 95 +++++++++++++++---- packages/nx/src/utils/package-json.ts | 10 +- 4 files changed, 131 insertions(+), 28 deletions(-) diff --git a/.nxignore b/.nxignore index bee50f804d4ca..6672afdcd4027 100644 --- a/.nxignore +++ b/.nxignore @@ -1,2 +1,2 @@ nx-dev/**/jest.config.js -.next +.next \ No newline at end of file diff --git a/packages/nx/src/generators/utils/project-configuration.spec.ts b/packages/nx/src/generators/utils/project-configuration.spec.ts index 4b663db33f2b2..c96e4daaf0fa6 100644 --- a/packages/nx/src/generators/utils/project-configuration.spec.ts +++ b/packages/nx/src/generators/utils/project-configuration.spec.ts @@ -277,5 +277,57 @@ describe('project configuration', () => { root: 'proj', }); }); + + it('should handle reading + writing project configuration', () => { + writeJson(tree, 'proj/package.json', { + name: 'proj', + nx: {}, + }); + + const proj = readProjectConfiguration(tree, 'proj'); + expect(proj).toEqual({ + name: 'proj', + root: 'proj', + }); + + updateProjectConfiguration(tree, 'proj', { + name: 'proj', + root: 'proj', + sourceRoot: 'proj/src', + targets: { + build: { + command: 'echo "building"', + }, + }, + }); + + const updatedProj = readProjectConfiguration(tree, 'proj'); + expect(updatedProj).toEqual({ + name: 'proj', + root: 'proj', + sourceRoot: 'proj/src', + targets: { + build: { + command: 'echo "building"', + }, + }, + }); + + expect(tree.read('proj/package.json', 'utf-8')).toMatchInlineSnapshot(` + "{ + "name": "proj", + "nx": { + "sourceRoot": "proj/src", + "targets": { + "build": { + "command": "echo \\"building\\"" + } + } + } + } + " + `); + expect(tree.exists('proj/project.json')).toBeFalsy(); + }); }); }); diff --git a/packages/nx/src/generators/utils/project-configuration.ts b/packages/nx/src/generators/utils/project-configuration.ts index 3158ee93db28a..74bc8b312fe98 100644 --- a/packages/nx/src/generators/utils/project-configuration.ts +++ b/packages/nx/src/generators/utils/project-configuration.ts @@ -1,10 +1,7 @@ import { minimatch } from 'minimatch'; -import { basename, join, relative } from 'path'; +import { basename, dirname, join, relative } from 'path'; -import { - buildProjectConfigurationFromPackageJson, - getGlobPatternsFromPackageManagerWorkspaces, -} from '../../plugins/package-json'; +import { getGlobPatternsFromPackageManagerWorkspaces } from '../../plugins/package-json'; import { buildProjectFromProjectJson } from '../../plugins/project-json/build-nodes/project-json'; import { renamePropertyWithStableKeys } from '../../adapter/angular-json'; import { @@ -23,6 +20,7 @@ import { readJson, writeJson } from './json'; import { readNxJson } from './nx-json'; import type { Tree } from '../tree'; +import { toProjectName } from '../../config/to-project-name'; export { readNxJson, updateNxJson } from './nx-json'; @@ -82,17 +80,73 @@ export function updateProjectConfiguration( projectName: string, projectConfiguration: ProjectConfiguration ): void { + if ( + tree.exists(joinPathFragments(projectConfiguration.root, 'project.json')) + ) { + updateProjectConfigurationInProjectJson( + tree, + projectName, + projectConfiguration + ); + } else if ( + tree.exists(joinPathFragments(projectConfiguration.root, 'package.json')) + ) { + updateProjectConfigurationInPackageJson( + tree, + projectName, + projectConfiguration + ); + } else { + throw new Error( + `Cannot update Project ${projectName} at ${projectConfiguration.root}. It either doesn't exist yet, or may not use project.json for configuration. Use \`addProjectConfiguration()\` instead if you want to create a new project.` + ); + } +} + +function updateProjectConfigurationInPackageJson( + tree: Tree, + projectName: string, + projectConfiguration: ProjectConfiguration +) { + const packageJsonFile = joinPathFragments( + projectConfiguration.root, + 'package.json' + ); + + const packageJson = readJson(tree, packageJsonFile); + + if (packageJson.name === projectConfiguration.name ?? projectName) { + delete projectConfiguration.name; + } + + if ( + projectConfiguration.targets && + !Object.keys(projectConfiguration.targets).length + ) { + delete projectConfiguration.targets; + } + + packageJson.nx = { + ...packageJson.nx, + ...projectConfiguration, + root: undefined, + }; + + writeJson(tree, packageJsonFile, packageJson); +} + +function updateProjectConfigurationInProjectJson( + tree: Tree, + projectName: string, + projectConfiguration: ProjectConfiguration +) { const projectConfigFile = joinPathFragments( projectConfiguration.root, 'project.json' ); - if (!tree.exists(projectConfigFile)) { - throw new Error( - `Cannot update Project ${projectName} at ${projectConfiguration.root}. It either doesn't exist yet, or may not use project.json for configuration. Use \`addProjectConfiguration()\` instead if you want to create a new project.` - ); - } handleEmptyTargets(projectName, projectConfiguration); + writeJson(tree, projectConfigFile, { name: projectConfiguration.name ?? projectName, $schema: getRelativeProjectJsonSchemaPath(tree, projectConfiguration), @@ -215,21 +269,22 @@ function readAndCombineAllProjectConfigurations(tree: Tree): { ); } else if (basename(projectFile) === 'package.json') { const packageJson = readJson(tree, projectFile); - const config = buildProjectConfigurationFromPackageJson( - packageJson, - tree.root, - projectFile, - readNxJson(tree) - ); + + // We don't want to have all of the extra inferred stuff in here, as + // when generators update the project they shouldn't inline that stuff. + // so rather than using `buildProjectFromPackageJson` and stripping it out + // we are going to build the config manually. + const config = { + root: dirname(projectFile), + name: packageJson.name ?? toProjectName(projectFile), + ...packageJson.nx, + }; if (!rootMap[config.root]) { mergeProjectConfigurationIntoRootMap( rootMap, // Inferred targets, tags, etc don't show up when running generators // This is to help avoid running into issues when trying to update the workspace - { - name: config.name, - root: config.root, - }, + config, undefined, undefined, true diff --git a/packages/nx/src/utils/package-json.ts b/packages/nx/src/utils/package-json.ts index 00315999d5ad8..a08f12ad06287 100644 --- a/packages/nx/src/utils/package-json.ts +++ b/packages/nx/src/utils/package-json.ts @@ -1,7 +1,7 @@ import { existsSync } from 'fs'; import { dirname, join } from 'path'; import { - InputDefinition, + ProjectConfiguration, ProjectMetadata, TargetConfiguration, } from '../config/workspace-json-project-json'; @@ -13,12 +13,8 @@ import { getPackageManagerCommand, } from './package-manager'; -export interface NxProjectPackageJsonConfiguration { - name?: string; - implicitDependencies?: string[]; - tags?: string[]; - namedInputs?: { [inputName: string]: (string | InputDefinition)[] }; - targets?: Record; +export interface NxProjectPackageJsonConfiguration + extends Partial { includedScripts?: string[]; }