Skip to content

Commit 9b95b46

Browse files
authored
feat: add support for .corepack.env (#642)
1 parent b456268 commit 9b95b46

File tree

4 files changed

+296
-18
lines changed

4 files changed

+296
-18
lines changed

README.md

+9
Original file line numberDiff line numberDiff line change
@@ -297,6 +297,7 @@ same major line. Should you need to upgrade to a new major, use an explicit
297297
set to `1` to have the URL shown. By default, when Corepack is called
298298
explicitly (e.g. `corepack pnpm …`), it is set to `0`; when Corepack is called
299299
implicitly (e.g. `pnpm …`), it is set to `1`.
300+
The default value cannot be overridden in a `.corepack.env` file.
300301
When standard input is a TTY and no CI environment is detected, Corepack will
301302
ask for user input before starting the download.
302303

@@ -322,6 +323,14 @@ same major line. Should you need to upgrade to a new major, use an explicit
322323
project. This means that it will always use the system-wide package manager
323324
regardless of what is being specified in the project's `packageManager` field.
324325

326+
- `COREPACK_ENV_FILE` can be set to `0` to request Corepack to not attempt to
327+
load `.corepack.env`; it can be set to a path to specify a different env file.
328+
Only keys that start with `COREPACK_` and are not in the exception list
329+
(`COREPACK_ENABLE_DOWNLOAD_PROMPT` and `COREPACK_ENV_FILE` are ignored)
330+
will be taken into account.
331+
For Node.js 18.x users, this setting has no effect as that version doesn't
332+
support parsing of `.env` files.
333+
325334
- `COREPACK_HOME` can be set in order to define where Corepack should install
326335
the package managers. By default it is set to `%LOCALAPPDATA%\node\corepack`
327336
on Windows, and to `$HOME/.cache/node/corepack` everywhere else.

sources/specUtils.ts

+46-2
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,14 @@ import path from 'path';
44
import semverSatisfies from 'semver/functions/satisfies';
55
import semverValid from 'semver/functions/valid';
66
import semverValidRange from 'semver/ranges/valid';
7+
import {parseEnv} from 'util';
78

89
import {PreparedPackageManagerInfo} from './Engine';
910
import * as debugUtils from './debugUtils';
1011
import {NodeError} from './nodeUtils';
1112
import * as nodeUtils from './nodeUtils';
1213
import {Descriptor, isSupportedPackageManager} from './types';
14+
import type {LocalEnvFile} from './types';
1315

1416
const nodeModulesRegExp = /[\\/]node_modules[\\/](@[^\\/]*[\\/])?([^@\\/][^\\/]*)$/;
1517

@@ -145,10 +147,17 @@ export async function setLocalPackageManager(cwd: string, info: PreparedPackageM
145147
};
146148
}
147149

150+
interface FoundSpecResult {
151+
type: `Found`;
152+
target: string;
153+
getSpec: () => Descriptor;
154+
range?: Descriptor & {onFail?: DevEngineDependency['onFail']};
155+
envFilePath?: string;
156+
}
148157
export type LoadSpecResult =
149158
| {type: `NoProject`, target: string}
150159
| {type: `NoSpec`, target: string}
151-
| {type: `Found`, target: string, getSpec: () => Descriptor, range?: Descriptor & {onFail?: DevEngineDependency['onFail']}};
160+
| FoundSpecResult;
152161

153162
export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
154163
let nextCwd = initialCwd;
@@ -157,6 +166,8 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
157166
let selection: {
158167
data: any;
159168
manifestPath: string;
169+
envFilePath?: string;
170+
localEnv: LocalEnvFile;
160171
} | null = null;
161172

162173
while (nextCwd !== currCwd && (!selection || !selection.data.packageManager)) {
@@ -184,12 +195,44 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
184195
if (typeof data !== `object` || data === null)
185196
throw new UsageError(`Invalid package.json in ${path.relative(initialCwd, manifestPath)}`);
186197

187-
selection = {data, manifestPath};
198+
let localEnv: LocalEnvFile;
199+
const envFilePath = path.resolve(currCwd, process.env.COREPACK_ENV_FILE ?? `.corepack.env`);
200+
if (process.env.COREPACK_ENV_FILE == `0`) {
201+
debugUtils.log(`Skipping env file as configured with COREPACK_ENV_FILE`);
202+
localEnv = process.env;
203+
} else if (typeof parseEnv !== `function`) {
204+
// TODO: remove this block when support for Node.js 18.x is dropped.
205+
debugUtils.log(`Skipping env file as it is not supported by the current version of Node.js`);
206+
localEnv = process.env;
207+
} else {
208+
debugUtils.log(`Checking ${envFilePath}`);
209+
try {
210+
localEnv = {
211+
...Object.fromEntries(Object.entries(parseEnv(await fs.promises.readFile(envFilePath, `utf8`))).filter(e => e[0].startsWith(`COREPACK_`))),
212+
...process.env,
213+
};
214+
debugUtils.log(`Successfully loaded env file found at ${envFilePath}`);
215+
} catch (err) {
216+
if ((err as NodeError)?.code !== `ENOENT`)
217+
throw err;
218+
219+
debugUtils.log(`No env file found at ${envFilePath}`);
220+
localEnv = process.env;
221+
}
222+
}
223+
224+
selection = {data, manifestPath, localEnv, envFilePath};
188225
}
189226

190227
if (selection === null)
191228
return {type: `NoProject`, target: path.join(initialCwd, `package.json`)};
192229

230+
let envFilePath: string | undefined;
231+
if (selection.localEnv !== process.env) {
232+
envFilePath = selection.envFilePath;
233+
process.env = selection.localEnv;
234+
}
235+
193236
const rawPmSpec = parsePackageJSON(selection.data);
194237
if (typeof rawPmSpec === `undefined`)
195238
return {type: `NoSpec`, target: selection.manifestPath};
@@ -199,6 +242,7 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
199242
return {
200243
type: `Found`,
201244
target: selection.manifestPath,
245+
envFilePath,
202246
range: selection.data.devEngines?.packageManager?.version && {
203247
name: selection.data.devEngines.packageManager.name,
204248
range: selection.data.devEngines.packageManager.version,

sources/types.ts

+2
Original file line numberDiff line numberDiff line change
@@ -160,3 +160,5 @@ export interface LazyLocator {
160160
*/
161161
reference: () => Promise<string>;
162162
}
163+
164+
export type LocalEnvFile = Record<string, string | undefined>;

0 commit comments

Comments
 (0)