Skip to content

Commit 0329cad

Browse files
authored
fix(core): Update move/remove workspace generators to work with ts project references (#29331)
<!-- Please make sure you have read the submission guidelines before posting an PR --> <!-- https://github.com/nrwl/nx/blob/master/CONTRIBUTING.md#-submitting-a-pr --> <!-- Please make sure that your commit message follows our format --> <!-- Example: `fix(nx): must begin with lowercase` --> <!-- If this is a particularly complex change or feature addition, you can request a dedicated Nx release for this pull request branch. Mention someone from the Nx team or the `@nrwl/nx-pipelines-reviewers` and they will confirm if the PR warrants its own release for testing purposes, and generate it for you if appropriate. --> ## Current Behavior <!-- This is the behavior we have today --> When a workspace is setup to use ts project references the move/remove generators from `@nx/workspace` do not work correctly or result in an incorrect state for the workspace. ## Expected Behavior <!-- This is the behavior we should expect with the changes in this PR --> This should work OOTB. ## Related Issue(s) <!-- Please link the issue being fixed so it gets closed when this is merged. --> Fixes #
1 parent 0888977 commit 0329cad

File tree

5 files changed

+313
-30
lines changed

5 files changed

+313
-30
lines changed

packages/workspace/src/generators/move/lib/update-imports.spec.ts

+85
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import { createTreeWithEmptyWorkspace } from '@nx/devkit/testing';
1010
import { Schema } from '../schema';
1111
import { normalizeSchema } from './normalize-schema';
1212
import { updateImports } from './update-imports';
13+
import * as tsSolution from '../../../utilities/typescript/ts-solution-setup';
1314

1415
// nx-ignore-next-line
1516
const { libraryGenerator } = require('@nx/js');
@@ -530,4 +531,88 @@ export MyExtendedClass extends MyClass {};`
530531
'@proj/my-source/server': ['my-destination/src/server.ts'],
531532
});
532533
});
534+
535+
describe('TypeScript project references', () => {
536+
beforeEach(() => {
537+
jest.spyOn(tsSolution, 'isUsingTsSolutionSetup').mockReturnValue(true);
538+
const tsconfigContent = {
539+
extends: './tsconfig.base.json',
540+
...readJson(tree, 'tsconfig.base.json'),
541+
};
542+
tree.write('tsconfig.json', JSON.stringify(tsconfigContent, null, 2));
543+
544+
const packageJson = readJson(tree, 'package.json');
545+
packageJson.workspaces = ['packages/**'];
546+
tree.write('package.json', JSON.stringify(packageJson, null, 2));
547+
});
548+
it('should work with updateImportPath=false', async () => {
549+
await libraryGenerator(tree, {
550+
directory: 'packages/my-source',
551+
});
552+
553+
const projectConfig = readProjectConfiguration(tree, 'my-source');
554+
555+
const tsconfigJson = readJson(tree, 'tsconfig.json');
556+
tsconfigJson.references = [{ path: './packages/my-source' }];
557+
tree.write('tsconfig.json', JSON.stringify(tsconfigJson, null, 2));
558+
559+
updateImports(
560+
tree,
561+
await normalizeSchema(
562+
tree,
563+
{
564+
...schema,
565+
updateImportPath: false,
566+
},
567+
projectConfig
568+
),
569+
570+
projectConfig
571+
);
572+
573+
expect(readJson(tree, 'tsconfig.json').references).toMatchInlineSnapshot(`
574+
[
575+
{
576+
"path": "./packages/my-source",
577+
},
578+
{
579+
"path": "./my-destination",
580+
},
581+
]
582+
`);
583+
});
584+
585+
it('should work with updateImportPath=true', async () => {
586+
await libraryGenerator(tree, {
587+
directory: 'packages/my-source',
588+
});
589+
590+
const projectConfig = readProjectConfiguration(tree, 'my-source');
591+
592+
const tsconfigJson = readJson(tree, 'tsconfig.json');
593+
tsconfigJson.references = [{ path: './packages/my-source' }];
594+
tree.write('tsconfig.json', JSON.stringify(tsconfigJson, null, 2));
595+
596+
updateImports(
597+
tree,
598+
await normalizeSchema(
599+
tree,
600+
{
601+
...schema,
602+
},
603+
projectConfig
604+
),
605+
606+
projectConfig
607+
);
608+
609+
expect(readJson(tree, 'tsconfig.json').references).toMatchInlineSnapshot(`
610+
[
611+
{
612+
"path": "./my-destination",
613+
},
614+
]
615+
`);
616+
});
617+
});
533618
});

packages/workspace/src/generators/move/lib/update-imports.ts

+82-21
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ import {
1111
visitNotIgnoredFiles,
1212
writeJson,
1313
} from '@nx/devkit';
14-
import { relative } from 'path';
14+
import { isAbsolute, normalize, relative } from 'path';
1515
import type * as ts from 'typescript';
1616
import { getImportPath } from '../../../utilities/get-import-path';
1717
import {
@@ -21,6 +21,7 @@ import {
2121
import { ensureTypescript } from '../../../utilities/typescript';
2222
import { NormalizedSchema } from '../schema';
2323
import { normalizePathSlashes } from './utils';
24+
import { isUsingTsSolutionSetup } from '../../../utilities/typescript/ts-solution-setup';
2425

2526
let tsModule: typeof import('typescript');
2627

@@ -41,9 +42,14 @@ export function updateImports(
4142
const { libsDir } = getWorkspaceLayout(tree);
4243
const projects = getProjects(tree);
4344

45+
const isUsingTsSolution = isUsingTsSolutionSetup(tree);
46+
4447
// use the source root to find the from location
4548
// this attempts to account for libs that have been created with --importPath
46-
const tsConfigPath = getRootTsConfigPathInTree(tree);
49+
const tsConfigPath = isUsingTsSolution
50+
? 'tsconfig.json'
51+
: getRootTsConfigPathInTree(tree);
52+
// If we are using a ts solution setup, we need to use tsconfig.json instead of tsconfig.base.json
4753
let tsConfig: any;
4854
let mainEntryPointImportPath: string;
4955
let secondaryEntryPointImportPaths: string[];
@@ -149,30 +155,85 @@ export function updateImports(
149155
};
150156

151157
if (tsConfig) {
152-
const path = tsConfig.compilerOptions.paths[projectRef.from] as string[];
153-
if (!path) {
154-
throw new Error(
155-
[
156-
`unable to find "${projectRef.from}" in`,
157-
`${tsConfigPath} compilerOptions.paths`,
158-
].join(' ')
158+
if (!isUsingTsSolution) {
159+
updateTsConfigPaths(
160+
tsConfig,
161+
projectRef,
162+
tsConfigPath,
163+
projectRoot,
164+
schema
159165
);
160-
}
161-
const updatedPath = path.map((x) =>
162-
joinPathFragments(projectRoot.to, relative(projectRoot.from, x))
163-
);
164-
165-
if (schema.updateImportPath && projectRef.to) {
166-
tsConfig.compilerOptions.paths[projectRef.to] = updatedPath;
167-
if (projectRef.from !== projectRef.to) {
168-
delete tsConfig.compilerOptions.paths[projectRef.from];
169-
}
170166
} else {
171-
tsConfig.compilerOptions.paths[projectRef.from] = updatedPath;
167+
updateTsConfigReferences(tsConfig, projectRoot, tsConfigPath, schema);
172168
}
169+
writeJson(tree, tsConfigPath, tsConfig);
173170
}
171+
}
172+
}
173+
174+
function updateTsConfigReferences(
175+
tsConfig: any,
176+
projectRoot: { from: string; to: string },
177+
tsConfigPath: string,
178+
schema: NormalizedSchema
179+
) {
180+
// Since paths can be './path' or 'path' we check if both are the same relative path to the workspace root
181+
const projectRefIndex = tsConfig.references.findIndex(
182+
(ref) => relative(ref.path, projectRoot.from) === ''
183+
);
184+
if (projectRefIndex === -1) {
185+
throw new Error(
186+
`unable to find "${projectRoot.from}" in ${tsConfigPath} references`
187+
);
188+
}
189+
const updatedPath = joinPathFragments(
190+
projectRoot.to,
191+
relative(projectRoot.from, tsConfig.references[projectRefIndex].path)
192+
);
193+
194+
let normalizedPath = normalize(updatedPath);
195+
if (
196+
!normalizedPath.startsWith('.') &&
197+
!normalizedPath.startsWith('../') &&
198+
!isAbsolute(normalizedPath)
199+
) {
200+
normalizedPath = `./${normalizedPath}`;
201+
}
174202

175-
writeJson(tree, tsConfigPath, tsConfig);
203+
if (schema.updateImportPath && projectRoot.to) {
204+
tsConfig.references[projectRefIndex].path = normalizedPath;
205+
} else {
206+
tsConfig.references.push({ path: normalizedPath });
207+
}
208+
}
209+
210+
function updateTsConfigPaths(
211+
tsConfig: any,
212+
projectRef: { from: string; to: string },
213+
tsConfigPath: string,
214+
projectRoot: { from: string; to: string },
215+
schema: NormalizedSchema
216+
) {
217+
const path = tsConfig.compilerOptions.paths[projectRef.from] as string[];
218+
if (!path) {
219+
throw new Error(
220+
[
221+
`unable to find "${projectRef.from}" in`,
222+
`${tsConfigPath} compilerOptions.paths`,
223+
].join(' ')
224+
);
225+
}
226+
const updatedPath = path.map((x) =>
227+
joinPathFragments(projectRoot.to, relative(projectRoot.from, x))
228+
);
229+
230+
if (schema.updateImportPath && projectRef.to) {
231+
tsConfig.compilerOptions.paths[projectRef.to] = updatedPath;
232+
if (projectRef.from !== projectRef.to) {
233+
delete tsConfig.compilerOptions.paths[projectRef.from];
234+
}
235+
} else {
236+
tsConfig.compilerOptions.paths[projectRef.from] = updatedPath;
176237
}
177238
}
178239

packages/workspace/src/generators/remove/lib/update-tsconfig.spec.ts

+41
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ import {
77
import { createTreeWithEmptyWorkspace } from '@nx/devkit/testing';
88
import { Schema } from '../schema';
99
import { updateTsconfig } from './update-tsconfig';
10+
import * as tsSolution from '../../../utilities/typescript/ts-solution-setup';
1011

1112
// nx-ignore-next-line
1213
const { libraryGenerator } = require('@nx/js');
@@ -212,4 +213,44 @@ describe('updateTsconfig', () => {
212213
'@proj/nested/whatever-name': ['libs/my-lib/nested-lib/src/index.ts'],
213214
});
214215
});
216+
217+
it('should work with tsSolution setup', async () => {
218+
jest.spyOn(tsSolution, 'isUsingTsSolutionSetup').mockReturnValue(true);
219+
220+
await libraryGenerator(tree, {
221+
directory: 'my-lib',
222+
});
223+
224+
const tsconfigContent = {
225+
extends: './tsconfig.base.json',
226+
compilerOptions: {},
227+
files: [],
228+
include: [],
229+
references: [
230+
{
231+
path: './my-lib',
232+
},
233+
],
234+
};
235+
236+
tree.write('tsconfig.json', JSON.stringify(tsconfigContent, null, 2));
237+
238+
graph = {
239+
nodes: {
240+
'my-lib': {
241+
name: 'my-lib',
242+
type: 'lib',
243+
data: {
244+
root: readProjectConfiguration(tree, 'my-lib').root,
245+
} as any,
246+
},
247+
},
248+
dependencies: {},
249+
};
250+
251+
await updateTsconfig(tree, schema);
252+
253+
const tsConfig = readJson(tree, 'tsconfig.json');
254+
expect(tsConfig.references).toEqual([]);
255+
});
215256
});

packages/workspace/src/generators/remove/lib/update-tsconfig.ts

+28-9
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import {
22
createProjectGraphAsync,
33
normalizePath,
44
ProjectGraph,
5+
readProjectsConfigurationFromProjectGraph,
56
Tree,
67
updateJson,
78
} from '@nx/devkit';
@@ -11,28 +12,46 @@ import {
1112
createProjectRootMappings,
1213
findProjectForPath,
1314
} from 'nx/src/project-graph/utils/find-project-for-path';
15+
import { isUsingTsSolutionSetup } from '../../../utilities/typescript/ts-solution-setup';
16+
import { relative } from 'path';
1417

1518
/**
1619
* Updates the tsconfig paths to remove the project.
1720
*
1821
* @param schema The options provided to the schematic
1922
*/
2023
export async function updateTsconfig(tree: Tree, schema: Schema) {
21-
const tsConfigPath = getRootTsConfigPathInTree(tree);
24+
const isUsingTsSolution = isUsingTsSolutionSetup(tree);
25+
const tsConfigPath = isUsingTsSolution
26+
? 'tsconfig.json'
27+
: getRootTsConfigPathInTree(tree);
2228

2329
if (tree.exists(tsConfigPath)) {
2430
const graph: ProjectGraph = await createProjectGraphAsync();
2531
const projectMapping = createProjectRootMappings(graph.nodes);
2632
updateJson(tree, tsConfigPath, (json) => {
27-
for (const importPath in json.compilerOptions.paths) {
28-
for (const path of json.compilerOptions.paths[importPath]) {
29-
const project = findProjectForPath(
30-
normalizePath(path),
31-
projectMapping
33+
if (isUsingTsSolution) {
34+
const projectConfigs = readProjectsConfigurationFromProjectGraph(graph);
35+
const project = projectConfigs.projects[schema.projectName];
36+
if (!project) {
37+
throw new Error(
38+
`Could not find project '${schema.project}'. Please choose a project that exists in the Nx Workspace.`
3239
);
33-
if (project === schema.projectName) {
34-
delete json.compilerOptions.paths[importPath];
35-
break;
40+
}
41+
json.references = json.references.filter(
42+
(ref) => relative(ref.path, project.root) !== ''
43+
);
44+
} else {
45+
for (const importPath in json.compilerOptions.paths) {
46+
for (const path of json.compilerOptions.paths[importPath]) {
47+
const project = findProjectForPath(
48+
normalizePath(path),
49+
projectMapping
50+
);
51+
if (project === schema.projectName) {
52+
delete json.compilerOptions.paths[importPath];
53+
break;
54+
}
3655
}
3756
}
3857
}

0 commit comments

Comments
 (0)