@@ -4,12 +4,14 @@ import path from 'path';
4
4
import semverSatisfies from 'semver/functions/satisfies' ;
5
5
import semverValid from 'semver/functions/valid' ;
6
6
import semverValidRange from 'semver/ranges/valid' ;
7
+ import { parseEnv } from 'util' ;
7
8
8
9
import { PreparedPackageManagerInfo } from './Engine' ;
9
10
import * as debugUtils from './debugUtils' ;
10
11
import { NodeError } from './nodeUtils' ;
11
12
import * as nodeUtils from './nodeUtils' ;
12
13
import { Descriptor , isSupportedPackageManager } from './types' ;
14
+ import type { LocalEnvFile } from './types' ;
13
15
14
16
const nodeModulesRegExp = / [ \\ / ] n o d e _ m o d u l e s [ \\ / ] ( @ [ ^ \\ / ] * [ \\ / ] ) ? ( [ ^ @ \\ / ] [ ^ \\ / ] * ) $ / ;
15
17
@@ -145,10 +147,17 @@ export async function setLocalPackageManager(cwd: string, info: PreparedPackageM
145
147
} ;
146
148
}
147
149
150
+ interface FoundSpecResult {
151
+ type : `Found`;
152
+ target : string ;
153
+ getSpec : ( ) => Descriptor ;
154
+ range ?: Descriptor & { onFail ?: DevEngineDependency [ 'onFail' ] } ;
155
+ envFilePath ?: string ;
156
+ }
148
157
export type LoadSpecResult =
149
158
| { type : `NoProject`, target : string }
150
159
| { type : `NoSpec`, target : string }
151
- | { type : `Found` , target : string , getSpec : ( ) => Descriptor , range ?: Descriptor & { onFail ?: DevEngineDependency [ 'onFail' ] } } ;
160
+ | FoundSpecResult ;
152
161
153
162
export async function loadSpec ( initialCwd : string ) : Promise < LoadSpecResult > {
154
163
let nextCwd = initialCwd ;
@@ -157,6 +166,8 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
157
166
let selection : {
158
167
data : any ;
159
168
manifestPath : string ;
169
+ envFilePath ?: string ;
170
+ localEnv : LocalEnvFile ;
160
171
} | null = null ;
161
172
162
173
while ( nextCwd !== currCwd && ( ! selection || ! selection . data . packageManager ) ) {
@@ -184,12 +195,44 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
184
195
if ( typeof data !== `object` || data === null )
185
196
throw new UsageError ( `Invalid package.json in ${ path . relative ( initialCwd , manifestPath ) } ` ) ;
186
197
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} ;
188
225
}
189
226
190
227
if ( selection === null )
191
228
return { type : `NoProject` , target : path . join ( initialCwd , `package.json` ) } ;
192
229
230
+ let envFilePath : string | undefined ;
231
+ if ( selection . localEnv !== process . env ) {
232
+ envFilePath = selection . envFilePath ;
233
+ process . env = selection . localEnv ;
234
+ }
235
+
193
236
const rawPmSpec = parsePackageJSON ( selection . data ) ;
194
237
if ( typeof rawPmSpec === `undefined` )
195
238
return { type : `NoSpec` , target : selection . manifestPath } ;
@@ -199,6 +242,7 @@ export async function loadSpec(initialCwd: string): Promise<LoadSpecResult> {
199
242
return {
200
243
type : `Found` ,
201
244
target : selection . manifestPath ,
245
+ envFilePath,
202
246
range : selection . data . devEngines ?. packageManager ?. version && {
203
247
name : selection . data . devEngines . packageManager . name ,
204
248
range : selection . data . devEngines . packageManager . version ,
0 commit comments