Skip to content

Commit c1eddbd

Browse files
committed
fix(@angular/cli): handle YARN_ environment variables during ng update and ng add
With this change we handle yarn specific environment variables during `ng update` and `ng add`. This is a follow up of #21297
1 parent f75a584 commit c1eddbd

File tree

4 files changed

+70
-28
lines changed

4 files changed

+70
-28
lines changed

packages/angular/cli/utilities/package-metadata.ts

+20-24
Original file line numberDiff line numberDiff line change
@@ -131,7 +131,7 @@ function readOptions(
131131
logger.info(`Locating potential ${baseFilename} files:`);
132132
}
133133

134-
let options: PackageManagerOptions = {};
134+
let rcOptions: PackageManagerOptions = {};
135135
for (const location of [...defaultConfigLocations, ...projectConfigLocations]) {
136136
if (existsSync(location)) {
137137
if (showPotentials) {
@@ -143,25 +143,33 @@ function readOptions(
143143
// See: https://github.com/npm/npm-registry-fetch/blob/ebddbe78a5f67118c1f7af2e02c8a22bcaf9e850/index.js#L99-L126
144144
const rcConfig: PackageManagerOptions = yarn ? lockfile.parse(data) : ini.parse(data);
145145

146-
options = normalizeOptions(rcConfig, location);
146+
rcOptions = normalizeOptions(rcConfig, location);
147147
}
148148
}
149149

150+
const envVariablesOptions: PackageManagerOptions = {};
150151
for (const [key, value] of Object.entries(process.env)) {
151-
if (!value || !key.toLowerCase().startsWith('npm_config_')) {
152+
if (!value) {
152153
continue;
153154
}
154155

155-
const normalizedName = key
156-
.substr(11)
157-
.replace(/(?!^)_/g, '-') // don't replace _ at the start of the key
158-
.toLowerCase();
159-
options[normalizedName] = value;
160-
}
156+
let normalizedName = key.toLowerCase();
157+
if (normalizedName.startsWith('npm_config_')) {
158+
normalizedName = normalizedName.substring(11);
159+
} else if (yarn && normalizedName.startsWith('yarn_')) {
160+
normalizedName = normalizedName.substring(5);
161+
} else {
162+
continue;
163+
}
161164

162-
options = normalizeOptions(options);
165+
normalizedName = normalizedName.replace(/(?!^)_/g, '-'); // don't replace _ at the start of the key.s
166+
envVariablesOptions[normalizedName] = value;
167+
}
163168

164-
return options;
169+
return {
170+
...rcOptions,
171+
...normalizeOptions(envVariablesOptions),
172+
};
165173
}
166174

167175
function normalizeOptions(
@@ -302,7 +310,6 @@ export async function fetchPackageManifest(
302310
} = {},
303311
): Promise<PackageManifest> {
304312
const { usingYarn = false, verbose = false, registry } = options;
305-
306313
ensureNpmrc(logger, usingYarn, verbose);
307314

308315
const response = await pacote.manifest(name, {
@@ -329,18 +336,7 @@ export function getNpmPackageJson(
329336
}
330337

331338
const { usingYarn = false, verbose = false, registry } = options;
332-
333-
if (!npmrc) {
334-
try {
335-
npmrc = readOptions(logger, false, verbose);
336-
} catch {}
337-
338-
if (usingYarn) {
339-
try {
340-
npmrc = { ...npmrc, ...readOptions(logger, true, verbose) };
341-
} catch {}
342-
}
343-
}
339+
ensureNpmrc(logger, usingYarn, verbose);
344340

345341
const resultPromise: Promise<NpmRepositoryPackageJson> = pacote.packument(packageName, {
346342
fullMetadata: true,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { expectFileNotToExist, expectFileToExist } from '../../../utils/fs';
2+
import { getActivePackageManager } from '../../../utils/packages';
3+
import { git, ng } from '../../../utils/process';
4+
import {
5+
createNpmConfigForAuthentication,
6+
setNpmEnvVarsForAuthentication,
7+
} from '../../../utils/registry';
8+
9+
export default async function () {
10+
// Yarn specific test that tests YARN_ env variables.
11+
// https://classic.yarnpkg.com/en/docs/envvars/
12+
if (getActivePackageManager() !== 'yarn') {
13+
return;
14+
}
15+
const command = ['add', '@angular/pwa', '--skip-confirmation'];
16+
17+
// Environment variables only
18+
await expectFileNotToExist('src/manifest.webmanifest');
19+
setNpmEnvVarsForAuthentication(false, true);
20+
await ng(...command);
21+
await expectFileToExist('src/manifest.webmanifest');
22+
await git('clean', '-dxf');
23+
24+
// Mix of config file and env vars works
25+
await expectFileNotToExist('src/manifest.webmanifest');
26+
await createNpmConfigForAuthentication(false, true);
27+
await ng(...command);
28+
await expectFileToExist('src/manifest.webmanifest');
29+
}

tests/legacy-cli/e2e/utils/registry.ts

+12-4
Original file line numberDiff line numberDiff line change
@@ -61,10 +61,18 @@ export function createNpmConfigForAuthentication(
6161
export function setNpmEnvVarsForAuthentication(
6262
/** When true, an incorrect token is used. Use this to validate authentication failures. */
6363
invalidToken = false,
64+
/** When true, `YARN_REGISTRY` is used instead of `NPM_CONFIG_REGISTRY`. */
65+
useYarnEnvVariable = false,
6466
): void {
65-
const token = invalidToken ? `invalid=` : VALID_TOKEN;
66-
const registry = SECURE_REGISTRY;
67+
delete process.env['YARN_REGISTRY'];
68+
delete process.env['NPM_CONFIG_REGISTRY'];
69+
70+
const registryKey = useYarnEnvVariable ? 'YARN_REGISTRY' : 'NPM_CONFIG_REGISTRY';
71+
process.env[registryKey] = `http:${SECURE_REGISTRY}`;
72+
73+
process.env['NPM_CONFIG__AUTH'] = invalidToken ? `invalid=` : VALID_TOKEN;
6774

68-
process.env['NPM_CONFIG_REGISTRY'] = `http:${registry}`;
69-
process.env['NPM_CONFIG__AUTH'] = token;
75+
// Needed for verdaccio when used with yarn
76+
// https://verdaccio.org/docs/en/cli-registry#yarn
77+
process.env['NPM_CONFIG_ALWAYS_AUTH'] = 'true';
7078
}

tests/legacy-cli/e2e_runner.ts

+9
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,10 @@ testsToRun
141141
const start = +new Date();
142142

143143
const module = require(absoluteName);
144+
const originalEnvVariables = {
145+
...process.env,
146+
};
147+
144148
const fn: (skipClean?: () => void) => Promise<void> | void =
145149
typeof module == 'function'
146150
? module
@@ -188,6 +192,11 @@ testsToRun
188192
);
189193
}
190194
})
195+
.finally(() => {
196+
// Restore env variables after each test.
197+
console.log(' Restoring original environment variables...');
198+
process.env = originalEnvVariables;
199+
})
191200
.then(
192201
() => printFooter(currentFileName, start),
193202
(err) => {

0 commit comments

Comments
 (0)