Skip to content

Commit 4955ee0

Browse files
committed
fix(@angular-devkit/schematics): properly resolve relative schematics when executed from a nested directory
Ensure that relative schematic paths are correctly resolved when invoked from within a nested directory. Closes: #29978
1 parent 441ba9a commit 4955ee0

File tree

2 files changed

+55
-8
lines changed

2 files changed

+55
-8
lines changed

packages/angular_devkit/schematics/tools/node-module-engine-host.ts

+2-8
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
*/
88

99
import { BaseException } from '@angular-devkit/core';
10-
import { dirname, join, resolve } from 'node:path';
10+
import { dirname, resolve } from 'node:path';
1111
import { RuleFactory } from '../src';
1212
import { FileSystemCollectionDesc, FileSystemSchematicDesc } from './description';
1313
import { ExportStringRef } from './export-ref';
@@ -46,20 +46,14 @@ export class NodeModulesEngineHost extends FileSystemEngineHostBase {
4646
}
4747
}
4848

49-
const relativeBase = requester ? dirname(requester) : process.cwd();
5049
let collectionPath: string | undefined = undefined;
51-
52-
if (name.startsWith('.')) {
53-
name = resolve(relativeBase, name);
54-
}
55-
5650
const resolveOptions = {
5751
paths: requester ? [dirname(requester), ...(this.paths || [])] : this.paths,
5852
};
5953

6054
// Try to resolve as a package
6155
try {
62-
const packageJsonPath = require.resolve(join(name, 'package.json'), resolveOptions);
56+
const packageJsonPath = require.resolve(`${name}/package.json`, resolveOptions);
6357
const { schematics } = require(packageJsonPath);
6458

6559
if (!schematics || typeof schematics !== 'string') {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
import assert from 'node:assert';
2+
import { join } from 'node:path';
3+
import { ng } from '../../utils/process';
4+
import { writeMultipleFiles, createDir } from '../../utils/fs';
5+
import { updateJsonFile } from '../../utils/project';
6+
7+
export default async function () {
8+
// setup temp collection
9+
await createDir('./fake-schematics');
10+
await writeMultipleFiles({
11+
'./fake-schematics/package.json': JSON.stringify({
12+
'schematics': './collection.json',
13+
}),
14+
'./fake-schematics/collection.json': JSON.stringify({
15+
'schematics': {
16+
'fake': {
17+
'description': 'Fake schematic',
18+
'schema': './fake-schema.json',
19+
'factory': './fake',
20+
},
21+
},
22+
}),
23+
'./fake-schematics/fake-schema.json': JSON.stringify({
24+
'$id': 'FakeSchema',
25+
'title': 'Fake Schema',
26+
'type': 'object',
27+
}),
28+
'./fake-schematics/fake.js': `
29+
exports.default = () => (host, context) => context.logger.info('fake schematic run.');
30+
`,
31+
});
32+
33+
await updateJsonFile('angular.json', (json) => {
34+
json.cli ??= {};
35+
json.cli.schematicCollections = ['./fake-schematics'];
36+
});
37+
38+
const { stdout: stdout1 } = await ng('generate', '--help');
39+
assert.match(stdout1, /Fake schematic/);
40+
41+
const { stdout: stdout2 } = await ng('generate', 'fake');
42+
assert.match(stdout2, /fake schematic run/);
43+
44+
// change cwd to a nested directory to validate the relative schematic is resolved correctly
45+
const originalCwd = process.cwd();
46+
try {
47+
process.chdir(join(originalCwd, 'src/app'));
48+
const { stdout: stdout3 } = await ng('generate', 'fake');
49+
assert.match(stdout3, /fake schematic run/);
50+
} finally {
51+
process.chdir(originalCwd);
52+
}
53+
}

0 commit comments

Comments
 (0)