@expo/env 2.2.2-canary-20260506-03817f5 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/constants.d.ts +13 -0
- package/build/constants.js +274 -0
- package/build/constants.js.map +1 -0
- package/build/index.d.ts +33 -0
- package/build/index.js +135 -4
- package/build/index.js.map +1 -1
- package/package.json +3 -3
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export declare function isUnsafeAllowedEnvKey(name: string): boolean;
|
|
2
|
+
export declare function isIgnoredEnvKey(name: string): boolean;
|
|
3
|
+
/**
|
|
4
|
+
* Whether a dotenv key represents per-developer/per-machine configuration that
|
|
5
|
+
* should only be loaded from `.local` env files (e.g. `.env.local`,
|
|
6
|
+
* `.env.development.local`). Committed `.env*` files cannot set these — that
|
|
7
|
+
* prevents a malicious project from redirecting developer-tool roots (e.g.
|
|
8
|
+
* `ANDROID_HOME`) via a supply-chain attack, while still letting developers
|
|
9
|
+
* pin them in their gitignored `.local` overrides.
|
|
10
|
+
*
|
|
11
|
+
* Honors `EXPO_UNSAFE_DOTENV_KEYS`: opt-in keys are allowed in any env file.
|
|
12
|
+
*/
|
|
13
|
+
export declare function isLocalEnvKey(name: string): boolean;
|
|
@@ -0,0 +1,274 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.isIgnoredEnvKey = isIgnoredEnvKey;
|
|
7
|
+
exports.isLocalEnvKey = isLocalEnvKey;
|
|
8
|
+
exports.isUnsafeAllowedEnvKey = isUnsafeAllowedEnvKey;
|
|
9
|
+
function _nodeOs() {
|
|
10
|
+
const data = _interopRequireDefault(require("node:os"));
|
|
11
|
+
_nodeOs = function () {
|
|
12
|
+
return data;
|
|
13
|
+
};
|
|
14
|
+
return data;
|
|
15
|
+
}
|
|
16
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
17
|
+
const platform = _nodeOs().default.platform();
|
|
18
|
+
|
|
19
|
+
// WARN(@kitten): We don't read this dynamically to ignore later modifications to this env var
|
|
20
|
+
const safeKeys = new Set(process.env.EXPO_UNSAFE_DOTENV_KEYS?.split(',').filter(x => !!x));
|
|
21
|
+
function isUnsafeAllowedEnvKey(name) {
|
|
22
|
+
return safeKeys.has(name);
|
|
23
|
+
}
|
|
24
|
+
function isIgnoredEnvKey(name) {
|
|
25
|
+
if (platform === 'darwin' && name.startsWith('DYLD_')) {
|
|
26
|
+
return true;
|
|
27
|
+
} else if (platform === 'linux' && name.startsWith('LD_')) {
|
|
28
|
+
return true;
|
|
29
|
+
} else if (safeKeys.has(name)) {
|
|
30
|
+
return false;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// NOTE(@kitten): Per-developer tool roots (ANDROID_HOME, JDK_HOME, DEVELOPER_DIR,
|
|
34
|
+
// npm/pnpm/yarn/bun paths, etc) are not blocked here — see `isLocalEnvKey`, which
|
|
35
|
+
// restricts them to `.local` env files (gitignored by convention) so committed
|
|
36
|
+
// `.env*` files cannot redirect them.
|
|
37
|
+
switch (name) {
|
|
38
|
+
// NOTE: Expo internal env vars
|
|
39
|
+
case '__EXPO_ENV_LOADED':
|
|
40
|
+
case 'EXPO_NO_DOTENV':
|
|
41
|
+
case 'EXPO_UNSAFE_DOTENV_KEYS':
|
|
42
|
+
return true;
|
|
43
|
+
|
|
44
|
+
// Linux dynamic-loader, can cause inconsistent calls
|
|
45
|
+
case 'LD_PRELOAD':
|
|
46
|
+
case 'LD_LIBRARY_PATH':
|
|
47
|
+
case 'LD_AUDIT':
|
|
48
|
+
return true;
|
|
49
|
+
|
|
50
|
+
// macOS dynamic-loader, can cause inconsistent calls
|
|
51
|
+
case 'DYLD_INSERT_LIBRARIES':
|
|
52
|
+
case 'DYLD_LIBRARY_PATH':
|
|
53
|
+
case 'DYLD_FRAMEWORK_PATH':
|
|
54
|
+
case 'DYLD_FALLBACK_LIBRARY_PATH':
|
|
55
|
+
case 'DYLD_FALLBACK_FRAMEWORK_PATH':
|
|
56
|
+
return true;
|
|
57
|
+
|
|
58
|
+
// OpenSSL
|
|
59
|
+
case 'SSLKEYLOGFILE':
|
|
60
|
+
return true;
|
|
61
|
+
|
|
62
|
+
// Changes Node behaviour and shouldn't be set in dotenv
|
|
63
|
+
case 'NODE_PATH':
|
|
64
|
+
case 'NODE_OPTIONS':
|
|
65
|
+
case 'NODE_EXTRA_CA_CERTS':
|
|
66
|
+
case 'NODE_TLS_REJECT_UNAUTHORIZED':
|
|
67
|
+
case 'NODE_COMPILE_CACHE':
|
|
68
|
+
case 'NPM_CONFIG_NODE_OPTIONS':
|
|
69
|
+
case 'NODE_REPL_EXTERNAL_MODULE':
|
|
70
|
+
return true;
|
|
71
|
+
|
|
72
|
+
// Changes Bun behaviour and shouldn't be set in dotenv
|
|
73
|
+
case 'BUN_RUNTIME_TRANSPILER_CACHE_PATH':
|
|
74
|
+
return true;
|
|
75
|
+
|
|
76
|
+
// Shell startup hooks
|
|
77
|
+
case 'BASH_ENV':
|
|
78
|
+
case 'ENV':
|
|
79
|
+
case 'ZDOTDIR':
|
|
80
|
+
case 'IFS':
|
|
81
|
+
case 'CDPATH':
|
|
82
|
+
case 'PROMPT_COMMAND':
|
|
83
|
+
case 'SHELLOPTS':
|
|
84
|
+
case 'BASHOPTS':
|
|
85
|
+
return true;
|
|
86
|
+
|
|
87
|
+
// Special git/ssh/gpg args
|
|
88
|
+
case 'GIT_SSH':
|
|
89
|
+
case 'GIT_SSH_COMMAND':
|
|
90
|
+
case 'GPG_TTY':
|
|
91
|
+
case 'SSH_ASKPASS':
|
|
92
|
+
case 'GIT_ASKPASS':
|
|
93
|
+
case 'GIT_EXEC_PATH':
|
|
94
|
+
return true;
|
|
95
|
+
|
|
96
|
+
// Perl libs
|
|
97
|
+
case 'PERL5OPT':
|
|
98
|
+
case 'PERL5LIB':
|
|
99
|
+
case 'PERLLIB':
|
|
100
|
+
return true;
|
|
101
|
+
|
|
102
|
+
// Python modules
|
|
103
|
+
case 'PYTHONSTARTUP':
|
|
104
|
+
case 'PYTHONPATH':
|
|
105
|
+
case 'PYTHONHOME':
|
|
106
|
+
case 'PYTHONINSPECT':
|
|
107
|
+
case 'PYTHONUSERBASE':
|
|
108
|
+
case 'PYTHONEXECUTABLE':
|
|
109
|
+
case 'PYTHONSAFEPATH':
|
|
110
|
+
case 'PYTJONNOUSERSITE':
|
|
111
|
+
return true;
|
|
112
|
+
|
|
113
|
+
// Ruby libs
|
|
114
|
+
case 'RUBYOPT':
|
|
115
|
+
case 'RUBYLIB':
|
|
116
|
+
case 'BUNDLE_GEMFILE':
|
|
117
|
+
case 'RUBYSHELL':
|
|
118
|
+
case 'RUBYPATH':
|
|
119
|
+
case 'GEM_HOME':
|
|
120
|
+
case 'GEM_PATH':
|
|
121
|
+
case 'BUNDLE_PATH':
|
|
122
|
+
return true;
|
|
123
|
+
|
|
124
|
+
// Java vars
|
|
125
|
+
case '_JAVA_OPTIONS':
|
|
126
|
+
case 'JAVA_TOOL_OPTIONS':
|
|
127
|
+
case 'JDK_JAVA_OPTIONS':
|
|
128
|
+
case 'CLASSPATH':
|
|
129
|
+
return true;
|
|
130
|
+
|
|
131
|
+
// User env vars
|
|
132
|
+
case 'HOME':
|
|
133
|
+
case 'USERPROFILE':
|
|
134
|
+
case 'HOMEDRIVE':
|
|
135
|
+
case 'HOMEPATH':
|
|
136
|
+
case 'TMPDIR':
|
|
137
|
+
case 'TMP':
|
|
138
|
+
case 'TEMP':
|
|
139
|
+
case 'USER':
|
|
140
|
+
case 'SHELL':
|
|
141
|
+
case 'PATH':
|
|
142
|
+
case 'PATHEXT':
|
|
143
|
+
case 'LANG':
|
|
144
|
+
case 'PWD':
|
|
145
|
+
case 'OLDPWD':
|
|
146
|
+
case 'TERMINFO':
|
|
147
|
+
return true;
|
|
148
|
+
|
|
149
|
+
// Windows-owned
|
|
150
|
+
case 'SYSTEMROOT':
|
|
151
|
+
case 'SystemRoot':
|
|
152
|
+
return true;
|
|
153
|
+
|
|
154
|
+
// User tools
|
|
155
|
+
case 'EDITOR':
|
|
156
|
+
case 'VISUAL':
|
|
157
|
+
case 'PAGER':
|
|
158
|
+
case 'MANPAGER':
|
|
159
|
+
return true;
|
|
160
|
+
|
|
161
|
+
// XDG dirs
|
|
162
|
+
case 'XDG_RUNTIME_DIR':
|
|
163
|
+
case 'XDG_STATE_HOME':
|
|
164
|
+
case 'XDG_DATA_HOME':
|
|
165
|
+
case 'XDG_CONFIG_DIRS':
|
|
166
|
+
case 'XDG_CACHE_HOME':
|
|
167
|
+
case 'XDG_CONFIG_HOME':
|
|
168
|
+
case 'XDG_BIN_HOME':
|
|
169
|
+
return true;
|
|
170
|
+
|
|
171
|
+
// direnv
|
|
172
|
+
case 'DIRENV_DIR':
|
|
173
|
+
case 'DIRENV_FILE':
|
|
174
|
+
case 'DIRENV_WATCHES':
|
|
175
|
+
case 'DIRENV_DIFF':
|
|
176
|
+
return true;
|
|
177
|
+
|
|
178
|
+
// Package-manager registry/install roots. No legitimate per-project `.env`
|
|
179
|
+
// use case — the established mechanism for each is a dedicated config file
|
|
180
|
+
// (`.npmrc`, `.yarnrc.yml`, `.bunfig.toml`) — and a malicious value is a
|
|
181
|
+
// supply-chain RCE the moment the CLI shells out to npm/yarn/pnpm/bun.
|
|
182
|
+
case 'NPM_CONFIG_REGISTRY':
|
|
183
|
+
case 'NPM_CONFIG_PREFIX':
|
|
184
|
+
case 'NPM_CONFIG_USERCONFIG':
|
|
185
|
+
case 'NPM_CONFIG_GLOBALCONFIG':
|
|
186
|
+
case 'NPM_CONFIG_CACHE':
|
|
187
|
+
case 'YARN_REGISTRY':
|
|
188
|
+
case 'YARN_CACHE_FOLDER':
|
|
189
|
+
case 'YARN_GLOBAL_FOLDER':
|
|
190
|
+
case 'PNPM_HOME':
|
|
191
|
+
case 'BUN_INSTALL':
|
|
192
|
+
case 'BUN_INSTALL_BIN':
|
|
193
|
+
case 'COCOAPODS_HOME':
|
|
194
|
+
case 'CMAKE_HOME':
|
|
195
|
+
return true;
|
|
196
|
+
default:
|
|
197
|
+
return false;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
/**
|
|
202
|
+
* Whether a dotenv key represents per-developer/per-machine configuration that
|
|
203
|
+
* should only be loaded from `.local` env files (e.g. `.env.local`,
|
|
204
|
+
* `.env.development.local`). Committed `.env*` files cannot set these — that
|
|
205
|
+
* prevents a malicious project from redirecting developer-tool roots (e.g.
|
|
206
|
+
* `ANDROID_HOME`) via a supply-chain attack, while still letting developers
|
|
207
|
+
* pin them in their gitignored `.local` overrides.
|
|
208
|
+
*
|
|
209
|
+
* Honors `EXPO_UNSAFE_DOTENV_KEYS`: opt-in keys are allowed in any env file.
|
|
210
|
+
*/
|
|
211
|
+
function isLocalEnvKey(name) {
|
|
212
|
+
if (safeKeys.has(name)) return false;
|
|
213
|
+
switch (name) {
|
|
214
|
+
// Android tooling
|
|
215
|
+
case 'ANDROID_HOME':
|
|
216
|
+
case 'ANDROID_SDK_ROOT':
|
|
217
|
+
case 'ANDROID_NDK_HOME':
|
|
218
|
+
case 'ANDROID_NDK_ROOT':
|
|
219
|
+
case 'ANDROID_AVD_HOME':
|
|
220
|
+
case 'ANDROID_EMULATOR_HOME':
|
|
221
|
+
case 'GRADLE_HOME':
|
|
222
|
+
case 'GRADLE_USER_HOME':
|
|
223
|
+
case 'KOTLIN_HOME':
|
|
224
|
+
return true;
|
|
225
|
+
|
|
226
|
+
// JVM tooling
|
|
227
|
+
case 'JAVA_HOME':
|
|
228
|
+
case 'JDK_HOME':
|
|
229
|
+
case 'JRE_HOME':
|
|
230
|
+
return true;
|
|
231
|
+
|
|
232
|
+
// Apple tooling
|
|
233
|
+
case 'DEVELOPER_DIR':
|
|
234
|
+
case 'XCODE_DEVELOPER_DIR_PATH':
|
|
235
|
+
return true;
|
|
236
|
+
|
|
237
|
+
// CocoaPods / Fastlane (secrets and non-exec config)
|
|
238
|
+
case 'COCOAPODS_DISABLE_STATS':
|
|
239
|
+
case 'FASTLANE_USER':
|
|
240
|
+
case 'FASTLANE_PASSWORD':
|
|
241
|
+
case 'FASTLANE_SESSION':
|
|
242
|
+
case 'FASTLANE_APPLE_APPLICATION_SPECIFIC_PASSWORD':
|
|
243
|
+
return true;
|
|
244
|
+
|
|
245
|
+
// Android NDK (per-project NDK version pinning is common)
|
|
246
|
+
case 'NDK_HOME':
|
|
247
|
+
case 'NDK_ROOT':
|
|
248
|
+
return true;
|
|
249
|
+
|
|
250
|
+
// Per-developer preferences and per-machine setup
|
|
251
|
+
case 'BROWSER':
|
|
252
|
+
case 'BROWSER_ARGS':
|
|
253
|
+
case 'HTTP_PROXY':
|
|
254
|
+
case 'http_proxy':
|
|
255
|
+
case 'HTTPS_PROXY':
|
|
256
|
+
case 'https_proxy':
|
|
257
|
+
case 'ALL_PROXY':
|
|
258
|
+
case 'all_proxy':
|
|
259
|
+
case 'NO_PROXY':
|
|
260
|
+
case 'no_proxy':
|
|
261
|
+
case 'FTP_PROXY':
|
|
262
|
+
case 'ftp_proxy':
|
|
263
|
+
case 'SSL_CRT_FILE':
|
|
264
|
+
case 'SSL_KEY_FILE':
|
|
265
|
+
case 'REACT_NATIVE_PACKAGER_HOSTNAME':
|
|
266
|
+
return true;
|
|
267
|
+
// NOTE(@kitten): Used to override where hermesc is found, not safe to read from .env
|
|
268
|
+
case 'REACT_NATIVE_OVERRIDE_HERMES_DIR':
|
|
269
|
+
return true;
|
|
270
|
+
default:
|
|
271
|
+
return false;
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
//# sourceMappingURL=constants.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"constants.js","names":["_nodeOs","data","_interopRequireDefault","require","e","__esModule","default","platform","os","safeKeys","Set","process","env","EXPO_UNSAFE_DOTENV_KEYS","split","filter","x","isUnsafeAllowedEnvKey","name","has","isIgnoredEnvKey","startsWith","isLocalEnvKey"],"sources":["../src/constants.ts"],"sourcesContent":["import os from 'node:os';\n\nconst platform = os.platform();\n\n// WARN(@kitten): We don't read this dynamically to ignore later modifications to this env var\nconst safeKeys = new Set(process.env.EXPO_UNSAFE_DOTENV_KEYS?.split(',').filter((x) => !!x));\n\nexport function isUnsafeAllowedEnvKey(name: string): boolean {\n return safeKeys.has(name);\n}\n\nexport function isIgnoredEnvKey(name: string) {\n if (platform === 'darwin' && name.startsWith('DYLD_')) {\n return true;\n } else if (platform === 'linux' && name.startsWith('LD_')) {\n return true;\n } else if (safeKeys.has(name)) {\n return false;\n }\n\n // NOTE(@kitten): Per-developer tool roots (ANDROID_HOME, JDK_HOME, DEVELOPER_DIR,\n // npm/pnpm/yarn/bun paths, etc) are not blocked here — see `isLocalEnvKey`, which\n // restricts them to `.local` env files (gitignored by convention) so committed\n // `.env*` files cannot redirect them.\n switch (name) {\n // NOTE: Expo internal env vars\n case '__EXPO_ENV_LOADED':\n case 'EXPO_NO_DOTENV':\n case 'EXPO_UNSAFE_DOTENV_KEYS':\n return true;\n\n // Linux dynamic-loader, can cause inconsistent calls\n case 'LD_PRELOAD':\n case 'LD_LIBRARY_PATH':\n case 'LD_AUDIT':\n return true;\n\n // macOS dynamic-loader, can cause inconsistent calls\n case 'DYLD_INSERT_LIBRARIES':\n case 'DYLD_LIBRARY_PATH':\n case 'DYLD_FRAMEWORK_PATH':\n case 'DYLD_FALLBACK_LIBRARY_PATH':\n case 'DYLD_FALLBACK_FRAMEWORK_PATH':\n return true;\n\n // OpenSSL\n case 'SSLKEYLOGFILE':\n return true;\n\n // Changes Node behaviour and shouldn't be set in dotenv\n case 'NODE_PATH':\n case 'NODE_OPTIONS':\n case 'NODE_EXTRA_CA_CERTS':\n case 'NODE_TLS_REJECT_UNAUTHORIZED':\n case 'NODE_COMPILE_CACHE':\n case 'NPM_CONFIG_NODE_OPTIONS':\n case 'NODE_REPL_EXTERNAL_MODULE':\n return true;\n\n // Changes Bun behaviour and shouldn't be set in dotenv\n case 'BUN_RUNTIME_TRANSPILER_CACHE_PATH':\n return true;\n\n // Shell startup hooks\n case 'BASH_ENV':\n case 'ENV':\n case 'ZDOTDIR':\n case 'IFS':\n case 'CDPATH':\n case 'PROMPT_COMMAND':\n case 'SHELLOPTS':\n case 'BASHOPTS':\n return true;\n\n // Special git/ssh/gpg args\n case 'GIT_SSH':\n case 'GIT_SSH_COMMAND':\n case 'GPG_TTY':\n case 'SSH_ASKPASS':\n case 'GIT_ASKPASS':\n case 'GIT_EXEC_PATH':\n return true;\n\n // Perl libs\n case 'PERL5OPT':\n case 'PERL5LIB':\n case 'PERLLIB':\n return true;\n\n // Python modules\n case 'PYTHONSTARTUP':\n case 'PYTHONPATH':\n case 'PYTHONHOME':\n case 'PYTHONINSPECT':\n case 'PYTHONUSERBASE':\n case 'PYTHONEXECUTABLE':\n case 'PYTHONSAFEPATH':\n case 'PYTJONNOUSERSITE':\n return true;\n\n // Ruby libs\n case 'RUBYOPT':\n case 'RUBYLIB':\n case 'BUNDLE_GEMFILE':\n case 'RUBYSHELL':\n case 'RUBYPATH':\n case 'GEM_HOME':\n case 'GEM_PATH':\n case 'BUNDLE_PATH':\n return true;\n\n // Java vars\n case '_JAVA_OPTIONS':\n case 'JAVA_TOOL_OPTIONS':\n case 'JDK_JAVA_OPTIONS':\n case 'CLASSPATH':\n return true;\n\n // User env vars\n case 'HOME':\n case 'USERPROFILE':\n case 'HOMEDRIVE':\n case 'HOMEPATH':\n case 'TMPDIR':\n case 'TMP':\n case 'TEMP':\n case 'USER':\n case 'SHELL':\n case 'PATH':\n case 'PATHEXT':\n case 'LANG':\n case 'PWD':\n case 'OLDPWD':\n case 'TERMINFO':\n return true;\n\n // Windows-owned\n case 'SYSTEMROOT':\n case 'SystemRoot':\n return true;\n\n // User tools\n case 'EDITOR':\n case 'VISUAL':\n case 'PAGER':\n case 'MANPAGER':\n return true;\n\n // XDG dirs\n case 'XDG_RUNTIME_DIR':\n case 'XDG_STATE_HOME':\n case 'XDG_DATA_HOME':\n case 'XDG_CONFIG_DIRS':\n case 'XDG_CACHE_HOME':\n case 'XDG_CONFIG_HOME':\n case 'XDG_BIN_HOME':\n return true;\n\n // direnv\n case 'DIRENV_DIR':\n case 'DIRENV_FILE':\n case 'DIRENV_WATCHES':\n case 'DIRENV_DIFF':\n return true;\n\n // Package-manager registry/install roots. No legitimate per-project `.env`\n // use case — the established mechanism for each is a dedicated config file\n // (`.npmrc`, `.yarnrc.yml`, `.bunfig.toml`) — and a malicious value is a\n // supply-chain RCE the moment the CLI shells out to npm/yarn/pnpm/bun.\n case 'NPM_CONFIG_REGISTRY':\n case 'NPM_CONFIG_PREFIX':\n case 'NPM_CONFIG_USERCONFIG':\n case 'NPM_CONFIG_GLOBALCONFIG':\n case 'NPM_CONFIG_CACHE':\n case 'YARN_REGISTRY':\n case 'YARN_CACHE_FOLDER':\n case 'YARN_GLOBAL_FOLDER':\n case 'PNPM_HOME':\n case 'BUN_INSTALL':\n case 'BUN_INSTALL_BIN':\n case 'COCOAPODS_HOME':\n case 'CMAKE_HOME':\n return true;\n\n default:\n return false;\n }\n}\n\n/**\n * Whether a dotenv key represents per-developer/per-machine configuration that\n * should only be loaded from `.local` env files (e.g. `.env.local`,\n * `.env.development.local`). Committed `.env*` files cannot set these — that\n * prevents a malicious project from redirecting developer-tool roots (e.g.\n * `ANDROID_HOME`) via a supply-chain attack, while still letting developers\n * pin them in their gitignored `.local` overrides.\n *\n * Honors `EXPO_UNSAFE_DOTENV_KEYS`: opt-in keys are allowed in any env file.\n */\nexport function isLocalEnvKey(name: string): boolean {\n if (safeKeys.has(name)) return false;\n switch (name) {\n // Android tooling\n case 'ANDROID_HOME':\n case 'ANDROID_SDK_ROOT':\n case 'ANDROID_NDK_HOME':\n case 'ANDROID_NDK_ROOT':\n case 'ANDROID_AVD_HOME':\n case 'ANDROID_EMULATOR_HOME':\n case 'GRADLE_HOME':\n case 'GRADLE_USER_HOME':\n case 'KOTLIN_HOME':\n return true;\n\n // JVM tooling\n case 'JAVA_HOME':\n case 'JDK_HOME':\n case 'JRE_HOME':\n return true;\n\n // Apple tooling\n case 'DEVELOPER_DIR':\n case 'XCODE_DEVELOPER_DIR_PATH':\n return true;\n\n // CocoaPods / Fastlane (secrets and non-exec config)\n case 'COCOAPODS_DISABLE_STATS':\n case 'FASTLANE_USER':\n case 'FASTLANE_PASSWORD':\n case 'FASTLANE_SESSION':\n case 'FASTLANE_APPLE_APPLICATION_SPECIFIC_PASSWORD':\n return true;\n\n // Android NDK (per-project NDK version pinning is common)\n case 'NDK_HOME':\n case 'NDK_ROOT':\n return true;\n\n // Per-developer preferences and per-machine setup\n case 'BROWSER':\n case 'BROWSER_ARGS':\n case 'HTTP_PROXY':\n case 'http_proxy':\n case 'HTTPS_PROXY':\n case 'https_proxy':\n case 'ALL_PROXY':\n case 'all_proxy':\n case 'NO_PROXY':\n case 'no_proxy':\n case 'FTP_PROXY':\n case 'ftp_proxy':\n case 'SSL_CRT_FILE':\n case 'SSL_KEY_FILE':\n case 'REACT_NATIVE_PACKAGER_HOSTNAME':\n return true;\n // NOTE(@kitten): Used to override where hermesc is found, not safe to read from .env\n case 'REACT_NATIVE_OVERRIDE_HERMES_DIR':\n return true;\n\n default:\n return false;\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
package/build/index.d.ts
CHANGED
|
@@ -27,6 +27,7 @@ export declare function parseEnvFiles(envFiles: string[], { systemEnv, }?: {
|
|
|
27
27
|
}): {
|
|
28
28
|
env: EnvOutput;
|
|
29
29
|
files: string[];
|
|
30
|
+
sensitiveLoadedKeys: string[];
|
|
30
31
|
};
|
|
31
32
|
/**
|
|
32
33
|
* Parse all environment variables using the list of `.env*` files, and mutate the system environment with these variables.
|
|
@@ -46,6 +47,7 @@ export declare function loadEnvFiles(envFiles: string[], { force, silent, system
|
|
|
46
47
|
loaded: string[];
|
|
47
48
|
env: EnvOutput;
|
|
48
49
|
files: string[];
|
|
50
|
+
sensitiveLoadedKeys: string[];
|
|
49
51
|
result: "loaded";
|
|
50
52
|
};
|
|
51
53
|
/**
|
|
@@ -55,6 +57,7 @@ export declare function loadEnvFiles(envFiles: string[], { force, silent, system
|
|
|
55
57
|
export declare function parseProjectEnv(projectRoot: string, options?: Parameters<typeof getEnvFiles>[0] & Parameters<typeof parseEnvFiles>[1]): {
|
|
56
58
|
env: EnvOutput;
|
|
57
59
|
files: string[];
|
|
60
|
+
sensitiveLoadedKeys: string[];
|
|
58
61
|
};
|
|
59
62
|
/**
|
|
60
63
|
* Parse all environment variables using the detected list of `.env*` files from a project.
|
|
@@ -69,8 +72,36 @@ export declare function loadProjectEnv(projectRoot: string, options?: Parameters
|
|
|
69
72
|
loaded: string[];
|
|
70
73
|
env: EnvOutput;
|
|
71
74
|
files: string[];
|
|
75
|
+
sensitiveLoadedKeys: string[];
|
|
72
76
|
result: "loaded";
|
|
73
77
|
};
|
|
78
|
+
/**
|
|
79
|
+
* Get a fresh clone of the system environment with all `@expo/env`-applied
|
|
80
|
+
* mutations reverted to their pre-load values. The result is intended to be
|
|
81
|
+
* passed as the `env` option of `child_process.spawn` / `@expo/spawn-async`
|
|
82
|
+
* when a subprocess should observe the environment as it was before any
|
|
83
|
+
* `.env*` files were loaded — for example, when resolving SDK tooling paths
|
|
84
|
+
* that should not be influenced by project-controlled `.env` values.
|
|
85
|
+
*
|
|
86
|
+
* Allocates lazily: nothing is held until this function is called, and each
|
|
87
|
+
* call returns a new object so callers may mutate it freely.
|
|
88
|
+
*
|
|
89
|
+
* @param systemEnv The env to revert against; defaults to `process.env`.
|
|
90
|
+
*/
|
|
91
|
+
export declare function getOriginalEnv(systemEnv?: EnvOutput): EnvOutput;
|
|
92
|
+
/**
|
|
93
|
+
* Get the pre-load value of a single environment variable as recorded by
|
|
94
|
+
* `@expo/env`. Falls through to the value in `systemEnv` for keys that
|
|
95
|
+
* `@expo/env` never touched. O(1) and allocation-free, intended for read-sites
|
|
96
|
+
* that resolve filesystem paths or executables from a single env var.
|
|
97
|
+
*
|
|
98
|
+
* Honors `EXPO_UNSAFE_DOTENV_KEYS`: keys the caller has explicitly opted into
|
|
99
|
+
* via the escape hatch return their currently loaded value, not the original.
|
|
100
|
+
*
|
|
101
|
+
* @param key The environment variable to read.
|
|
102
|
+
* @param systemEnv The env to read against; defaults to `process.env`.
|
|
103
|
+
*/
|
|
104
|
+
export declare function getOriginalEnvValue(key: string, systemEnv?: EnvOutput): string | undefined;
|
|
74
105
|
/** Log the loaded environment info from the loaded results */
|
|
75
106
|
export declare function logLoadedEnv(envInfo: ReturnType<typeof loadEnvFiles>, options?: Parameters<typeof loadEnvFiles>[1]): {
|
|
76
107
|
result: "skipped";
|
|
@@ -79,6 +110,7 @@ export declare function logLoadedEnv(envInfo: ReturnType<typeof loadEnvFiles>, o
|
|
|
79
110
|
loaded: string[];
|
|
80
111
|
env: EnvOutput;
|
|
81
112
|
files: string[];
|
|
113
|
+
sensitiveLoadedKeys: string[];
|
|
82
114
|
result: "loaded";
|
|
83
115
|
};
|
|
84
116
|
/**
|
|
@@ -93,6 +125,7 @@ export declare function get(projectRoot: string, { force, silent, }?: {
|
|
|
93
125
|
}): {
|
|
94
126
|
env: EnvOutput;
|
|
95
127
|
files: string[];
|
|
128
|
+
sensitiveLoadedKeys: string[];
|
|
96
129
|
};
|
|
97
130
|
/**
|
|
98
131
|
* Load environment variables from .env files and mutate the current `process.env` with the results.
|
package/build/index.js
CHANGED
|
@@ -7,6 +7,8 @@ exports.LOADED_ENV_NAME = exports.KNOWN_MODES = void 0;
|
|
|
7
7
|
exports.get = get;
|
|
8
8
|
exports.getEnvFiles = getEnvFiles;
|
|
9
9
|
exports.getFiles = getFiles;
|
|
10
|
+
exports.getOriginalEnv = getOriginalEnv;
|
|
11
|
+
exports.getOriginalEnvValue = getOriginalEnvValue;
|
|
10
12
|
exports.isEnabled = isEnabled;
|
|
11
13
|
exports.load = load;
|
|
12
14
|
exports.loadEnvFiles = loadEnvFiles;
|
|
@@ -50,6 +52,13 @@ function _nodePath() {
|
|
|
50
52
|
};
|
|
51
53
|
return data;
|
|
52
54
|
}
|
|
55
|
+
function _constants() {
|
|
56
|
+
const data = require("./constants");
|
|
57
|
+
_constants = function () {
|
|
58
|
+
return data;
|
|
59
|
+
};
|
|
60
|
+
return data;
|
|
61
|
+
}
|
|
53
62
|
function _parse() {
|
|
54
63
|
const data = require("./parse");
|
|
55
64
|
_parse = function () {
|
|
@@ -59,6 +68,20 @@ function _parse() {
|
|
|
59
68
|
}
|
|
60
69
|
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
61
70
|
const debug = require('debug')('expo:env');
|
|
71
|
+
const ORIGINAL_ENV_BACKUP_KEY = Symbol.for('@expo/env.originalEnvBackup.v1');
|
|
72
|
+
const globalStore = globalThis;
|
|
73
|
+
const originalEnvBackup = globalStore[ORIGINAL_ENV_BACKUP_KEY] ?? (globalStore[ORIGINAL_ENV_BACKUP_KEY] = new WeakMap());
|
|
74
|
+
function rememberOriginal(systemEnv, key) {
|
|
75
|
+
if ((0, _constants().isUnsafeAllowedEnvKey)(key)) return;
|
|
76
|
+
let backup = originalEnvBackup.get(systemEnv);
|
|
77
|
+
if (!backup) {
|
|
78
|
+
backup = new Map();
|
|
79
|
+
originalEnvBackup.set(systemEnv, backup);
|
|
80
|
+
}
|
|
81
|
+
if (!backup.has(key)) {
|
|
82
|
+
backup.set(key, systemEnv[key]);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
62
85
|
|
|
63
86
|
/** Determine if the `.env` files are enabled or not, through `EXPO_NO_DOTENV` */
|
|
64
87
|
function isEnabled() {
|
|
@@ -114,7 +137,8 @@ function parseEnvFiles(envFiles, {
|
|
|
114
137
|
debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);
|
|
115
138
|
return {
|
|
116
139
|
env: {},
|
|
117
|
-
files: []
|
|
140
|
+
files: [],
|
|
141
|
+
sensitiveLoadedKeys: []
|
|
118
142
|
};
|
|
119
143
|
}
|
|
120
144
|
|
|
@@ -126,6 +150,9 @@ function parseEnvFiles(envFiles, {
|
|
|
126
150
|
// https://github.com/motdotla/dotenv-expand
|
|
127
151
|
const loadedEnvVars = {};
|
|
128
152
|
const loadedEnvFiles = [];
|
|
153
|
+
const blockedByFile = {};
|
|
154
|
+
const localOnlyByFile = {};
|
|
155
|
+
const sensitive = new Set();
|
|
129
156
|
|
|
130
157
|
// Iterate over each dotenv file in lowest prio to highest prio order.
|
|
131
158
|
// This step won't write to the process.env, but will overwrite the parsed envs.
|
|
@@ -133,9 +160,23 @@ function parseEnvFiles(envFiles, {
|
|
|
133
160
|
try {
|
|
134
161
|
const envFileContent = _nodeFs().default.readFileSync(envFile, 'utf8');
|
|
135
162
|
const envFileParsed = (0, _parse().parse)(envFileContent);
|
|
163
|
+
const isLocalFile = _nodePath().default.basename(envFile).endsWith('.local');
|
|
136
164
|
loadedEnvFiles.push(envFile);
|
|
137
165
|
debug(`Loaded environment variables from: ${envFile}`);
|
|
138
166
|
for (const key of Object.keys(envFileParsed)) {
|
|
167
|
+
if ((0, _constants().isIgnoredEnvKey)(key)) {
|
|
168
|
+
(blockedByFile[envFile] ||= []).push(key);
|
|
169
|
+
debug(`"${key}" is blocked from dotenv files, skipping in: ${envFile}`);
|
|
170
|
+
continue;
|
|
171
|
+
}
|
|
172
|
+
if (!isLocalFile && (0, _constants().isLocalEnvKey)(key)) {
|
|
173
|
+
(localOnlyByFile[envFile] ||= []).push(key);
|
|
174
|
+
debug(`"${key}" is only allowed in .local env files, skipping in: ${envFile}`);
|
|
175
|
+
continue;
|
|
176
|
+
}
|
|
177
|
+
if (isLocalFile && (0, _constants().isLocalEnvKey)(key)) {
|
|
178
|
+
sensitive.add(key);
|
|
179
|
+
}
|
|
139
180
|
if (typeof loadedEnvVars[key] !== 'undefined') {
|
|
140
181
|
debug(`"${key}" is already defined and overwritten by: ${envFile}`);
|
|
141
182
|
}
|
|
@@ -154,11 +195,35 @@ function parseEnvFiles(envFiles, {
|
|
|
154
195
|
throw error;
|
|
155
196
|
}
|
|
156
197
|
});
|
|
198
|
+
const violations = [];
|
|
199
|
+
if (Object.keys(blockedByFile).length > 0) {
|
|
200
|
+
violations.push(formatBlockedViolation(blockedByFile));
|
|
201
|
+
}
|
|
202
|
+
if (Object.keys(localOnlyByFile).length > 0) {
|
|
203
|
+
violations.push(formatLocalOnlyViolation(localOnlyByFile));
|
|
204
|
+
}
|
|
205
|
+
if (violations.length > 0) {
|
|
206
|
+
throw new Error(violations.join('\n\n'));
|
|
207
|
+
}
|
|
208
|
+
const env = (0, _parse().expand)(loadedEnvVars, systemEnv);
|
|
209
|
+
for (const key in env) {
|
|
210
|
+
rememberOriginal(systemEnv, key);
|
|
211
|
+
}
|
|
157
212
|
return {
|
|
158
|
-
env
|
|
159
|
-
files: loadedEnvFiles.reverse()
|
|
213
|
+
env,
|
|
214
|
+
files: loadedEnvFiles.reverse(),
|
|
215
|
+
sensitiveLoadedKeys: [...sensitive]
|
|
160
216
|
};
|
|
161
217
|
}
|
|
218
|
+
function formatViolationFiles(byFile) {
|
|
219
|
+
return Object.entries(byFile).map(([file, keys]) => ` ${_nodePath().default.basename(file)}: ${keys.join(', ')}`).join('\n');
|
|
220
|
+
}
|
|
221
|
+
function formatBlockedViolation(byFile) {
|
|
222
|
+
return ['Refused to load dangerous environment variables from .env files.', 'Opt in via EXPO_UNSAFE_DOTENV_KEYS in your shell environment if you truly need them.', '', formatViolationFiles(byFile)].join('\n');
|
|
223
|
+
}
|
|
224
|
+
function formatLocalOnlyViolation(byFile) {
|
|
225
|
+
return ['Refused to load personal environment variables from a non-.local env file.', 'Move them to a .local env file.', '', formatViolationFiles(byFile)].join('\n');
|
|
226
|
+
}
|
|
162
227
|
|
|
163
228
|
/**
|
|
164
229
|
* Parse all environment variables using the list of `.env*` files, and mutate the system environment with these variables.
|
|
@@ -185,12 +250,14 @@ function loadEnvFiles(envFiles, {
|
|
|
185
250
|
if (typeof systemEnv[key] !== 'undefined') {
|
|
186
251
|
debug(`"${key}" is already defined and IS NOT overwritten`);
|
|
187
252
|
} else {
|
|
253
|
+
rememberOriginal(systemEnv, key);
|
|
188
254
|
systemEnv[key] = parsed.env[key];
|
|
189
255
|
loadedEnvKeys.push(key);
|
|
190
256
|
}
|
|
191
257
|
}
|
|
192
258
|
|
|
193
259
|
// Mark the environment as loaded
|
|
260
|
+
rememberOriginal(systemEnv, LOADED_ENV_NAME);
|
|
194
261
|
systemEnv[LOADED_ENV_NAME] = JSON.stringify(loadedEnvKeys);
|
|
195
262
|
return {
|
|
196
263
|
result: 'loaded',
|
|
@@ -217,6 +284,56 @@ function loadProjectEnv(projectRoot, options) {
|
|
|
217
284
|
return loadEnvFiles(getEnvFiles(options).map(envFile => _nodePath().default.join(projectRoot, envFile)), options);
|
|
218
285
|
}
|
|
219
286
|
|
|
287
|
+
/**
|
|
288
|
+
* Get a fresh clone of the system environment with all `@expo/env`-applied
|
|
289
|
+
* mutations reverted to their pre-load values. The result is intended to be
|
|
290
|
+
* passed as the `env` option of `child_process.spawn` / `@expo/spawn-async`
|
|
291
|
+
* when a subprocess should observe the environment as it was before any
|
|
292
|
+
* `.env*` files were loaded — for example, when resolving SDK tooling paths
|
|
293
|
+
* that should not be influenced by project-controlled `.env` values.
|
|
294
|
+
*
|
|
295
|
+
* Allocates lazily: nothing is held until this function is called, and each
|
|
296
|
+
* call returns a new object so callers may mutate it freely.
|
|
297
|
+
*
|
|
298
|
+
* @param systemEnv The env to revert against; defaults to `process.env`.
|
|
299
|
+
*/
|
|
300
|
+
function getOriginalEnv(systemEnv = process.env) {
|
|
301
|
+
const result = {
|
|
302
|
+
...systemEnv
|
|
303
|
+
};
|
|
304
|
+
const backup = originalEnvBackup.get(systemEnv);
|
|
305
|
+
if (backup) {
|
|
306
|
+
for (const [key, original] of backup) {
|
|
307
|
+
if (original === undefined) {
|
|
308
|
+
delete result[key];
|
|
309
|
+
} else {
|
|
310
|
+
result[key] = original;
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
return result;
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
/**
|
|
318
|
+
* Get the pre-load value of a single environment variable as recorded by
|
|
319
|
+
* `@expo/env`. Falls through to the value in `systemEnv` for keys that
|
|
320
|
+
* `@expo/env` never touched. O(1) and allocation-free, intended for read-sites
|
|
321
|
+
* that resolve filesystem paths or executables from a single env var.
|
|
322
|
+
*
|
|
323
|
+
* Honors `EXPO_UNSAFE_DOTENV_KEYS`: keys the caller has explicitly opted into
|
|
324
|
+
* via the escape hatch return their currently loaded value, not the original.
|
|
325
|
+
*
|
|
326
|
+
* @param key The environment variable to read.
|
|
327
|
+
* @param systemEnv The env to read against; defaults to `process.env`.
|
|
328
|
+
*/
|
|
329
|
+
function getOriginalEnvValue(key, systemEnv = process.env) {
|
|
330
|
+
const backup = originalEnvBackup.get(systemEnv);
|
|
331
|
+
if (backup && backup.has(key)) {
|
|
332
|
+
return backup.get(key);
|
|
333
|
+
}
|
|
334
|
+
return systemEnv[key];
|
|
335
|
+
}
|
|
336
|
+
|
|
220
337
|
/** Log the loaded environment info from the loaded results */
|
|
221
338
|
function logLoadedEnv(envInfo, options = {}) {
|
|
222
339
|
// Skip when running in force mode, or no environment variables are loaded
|
|
@@ -229,6 +346,13 @@ function logLoadedEnv(envInfo, options = {}) {
|
|
|
229
346
|
|
|
230
347
|
// Log the loaded environment variables
|
|
231
348
|
_nodeConsole().default.log(_chalk().default.gray('env: export', envInfo.loaded.join(' ')));
|
|
349
|
+
|
|
350
|
+
// Highlight developer-tool roots / secrets that were loaded from a .local file —
|
|
351
|
+
// the same keys would be refused from any non-.local file. Surfacing them here
|
|
352
|
+
// tells the user which "sensitive" values are influencing the build.
|
|
353
|
+
if (envInfo.result === 'loaded' && envInfo.sensitiveLoadedKeys?.length) {
|
|
354
|
+
_nodeConsole().default.log(_chalk().default.yellow('env: export (sensitive)', envInfo.sensitiveLoadedKeys.join(' ')));
|
|
355
|
+
}
|
|
232
356
|
return envInfo;
|
|
233
357
|
}
|
|
234
358
|
|
|
@@ -250,7 +374,8 @@ function get(projectRoot, {
|
|
|
250
374
|
debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);
|
|
251
375
|
return {
|
|
252
376
|
env: {},
|
|
253
|
-
files: []
|
|
377
|
+
files: [],
|
|
378
|
+
sensitiveLoadedKeys: []
|
|
254
379
|
};
|
|
255
380
|
}
|
|
256
381
|
if (force || !memo) {
|
|
@@ -278,6 +403,7 @@ function load(projectRoot, options = {}) {
|
|
|
278
403
|
debug(`"${key}" is already defined and IS NOT overwritten`);
|
|
279
404
|
} else {
|
|
280
405
|
// Avoid creating a new object, mutate it instead as this causes problems in Bun
|
|
406
|
+
rememberOriginal(process.env, key);
|
|
281
407
|
process.env[key] = envInfo.env[key];
|
|
282
408
|
loadedEnvKeys.push(key);
|
|
283
409
|
}
|
|
@@ -318,6 +444,11 @@ function getFiles(mode, {
|
|
|
318
444
|
function parseEnv(contents, sourceEnv) {
|
|
319
445
|
try {
|
|
320
446
|
const env = (0, _parse().parse)(contents);
|
|
447
|
+
for (const key in env) {
|
|
448
|
+
if ((0, _constants().isIgnoredEnvKey)(key)) {
|
|
449
|
+
delete env[key];
|
|
450
|
+
}
|
|
451
|
+
}
|
|
321
452
|
return (0, _parse().expand)(env, sourceEnv || {});
|
|
322
453
|
} catch {
|
|
323
454
|
return {};
|
package/build/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":["_chalk","data","_interopRequireDefault","require","_getenv","_nodeConsole","_nodeFs","_nodePath","_parse","e","__esModule","default","debug","isEnabled","boolish","KNOWN_MODES","exports","LOADED_ENV_NAME","getEnvFiles","mode","process","env","NODE_ENV","silent","logError","console","error","logWarning","warn","includes","filter","Boolean","parseEnvFiles","envFiles","systemEnv","files","loadedEnvVars","loadedEnvFiles","reverse","forEach","envFile","envFileContent","fs","readFileSync","envFileParsed","parse","push","key","Object","keys","code","expand","loadEnvFiles","force","result","loaded","JSON","parsed","loadedEnvKeys","stringify","parseProjectEnv","projectRoot","options","map","path","join","loadProjectEnv","logLoadedEnv","envInfo","length","log","chalk","gray","file","basename","memo","get","load","getFiles","parseEnv","contents","sourceEnv"],"sources":["../src/index.ts"],"sourcesContent":["import chalk from 'chalk';\nimport { boolish } from 'getenv';\nimport console from 'node:console';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\nimport { parse, expand, type EnvOutput } from './parse';\n\nconst debug = require('debug')('expo:env') as typeof console.log;\n\n/** Determine if the `.env` files are enabled or not, through `EXPO_NO_DOTENV` */\nexport function isEnabled() {\n return !boolish('EXPO_NO_DOTENV', false);\n}\n\n/** All conventional modes that should not cause warnings */\nexport const KNOWN_MODES = ['development', 'test', 'production'];\n\n/** The environment variable name to use when marking the environment as loaded */\nexport const LOADED_ENV_NAME = '__EXPO_ENV_LOADED';\n\n/**\n * Get a list of all `.env*` files based on the `NODE_ENV` mode.\n * This returns a list of files, in order of highest priority to lowest priority.\n *\n * @see https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n */\nexport function getEnvFiles({\n mode = process.env.NODE_ENV,\n silent,\n}: {\n /** The mode to use when creating the list of `.env*` files, defaults to `NODE_ENV` */\n mode?: string;\n /** If possible misconfiguration warnings should be logged, or only logged as debug log */\n silent?: boolean;\n} = {}) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return [];\n }\n\n const logError = silent ? debug : console.error;\n const logWarning = silent ? debug : console.warn;\n\n if (!mode) {\n logError(\n `The NODE_ENV environment variable is required but was not specified. Ensure the project is bundled with Expo CLI or NODE_ENV is set. Using only .env.local and .env`\n );\n return ['.env.local', '.env'];\n }\n\n if (!KNOWN_MODES.includes(mode)) {\n logWarning(\n `NODE_ENV=\"${mode}\" is non-conventional and might cause development code to run in production. Use \"development\", \"test\", or \"production\" instead. Continuing with non-conventional mode`\n );\n }\n\n // see: https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n return [\n `.env.${mode}.local`,\n // Don't include `.env.local` for `test` environment\n // since normally you expect tests to produce the same\n // results for everyone\n mode !== 'test' && `.env.local`,\n `.env.${mode}`,\n `.env`,\n ].filter(Boolean) as string[];\n}\n\n/**\n * Parse all environment variables using the list of `.env*` files, in order of higest priority to lowest priority.\n * This does not check for collisions of existing system environment variables, or mutates the system environment variables.\n */\nexport function parseEnvFiles(\n envFiles: string[],\n {\n systemEnv = process.env,\n }: {\n /** The system environment to use when expanding environment variables, defaults to `process.env` */\n systemEnv?: EnvOutput;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return { env: {}, files: [] };\n }\n\n // Load environment variables from .env* files. Suppress warnings using silent\n // if this file is missing. Dotenv will only parse the environment variables,\n // `@expo/env` will set the resulting variables to the current process.\n // Variable expansion is supported in .env files, and executed as final step.\n // https://github.com/motdotla/dotenv\n // https://github.com/motdotla/dotenv-expand\n const loadedEnvVars: EnvOutput = {};\n const loadedEnvFiles: string[] = [];\n\n // Iterate over each dotenv file in lowest prio to highest prio order.\n // This step won't write to the process.env, but will overwrite the parsed envs.\n [...envFiles].reverse().forEach((envFile) => {\n try {\n const envFileContent = fs.readFileSync(envFile, 'utf8');\n const envFileParsed = parse(envFileContent);\n\n loadedEnvFiles.push(envFile);\n debug(`Loaded environment variables from: ${envFile}`);\n\n for (const key of Object.keys(envFileParsed)) {\n if (typeof loadedEnvVars[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and overwritten by: ${envFile}`);\n }\n\n loadedEnvVars[key] = envFileParsed[key];\n }\n } catch (error: any) {\n if ('code' in error && error.code === 'ENOENT') {\n return debug(`${envFile} does not exist, skipping this env file`);\n }\n if ('code' in error && error.code === 'EISDIR') {\n return debug(`${envFile} is a directory, skipping this env file`);\n }\n if ('code' in error && error.code === 'EACCES') {\n return debug(`No permission to read ${envFile}, skipping this env file`);\n }\n\n throw error;\n }\n });\n\n return {\n env: expand(loadedEnvVars, systemEnv),\n files: loadedEnvFiles.reverse(),\n };\n}\n\n/**\n * Parse all environment variables using the list of `.env*` files, and mutate the system environment with these variables.\n * This won't override existing environment variables defined in the system environment.\n * Once the mutations are done, this will also set a propert `__EXPO_ENV=true` on the system env to avoid multiple mutations.\n * This check can be disabled through `{ force: true }`.\n */\nexport function loadEnvFiles(\n envFiles: string[],\n {\n force,\n silent = false,\n systemEnv = process.env,\n }: Parameters<typeof parseEnvFiles>[1] & {\n /** If the environment variables should be applied to the system environment, regardless of previous mutations */\n force?: boolean;\n /** If possible misconfiguration warnings should be logged, or only logged as debug log */\n silent?: boolean;\n } = {}\n) {\n if (!force && systemEnv[LOADED_ENV_NAME]) {\n return { result: 'skipped' as const, loaded: JSON.parse(systemEnv[LOADED_ENV_NAME]) };\n }\n\n const parsed = parseEnvFiles(envFiles, { systemEnv });\n const loadedEnvKeys: string[] = [];\n\n for (const key in parsed.env) {\n if (typeof systemEnv[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and IS NOT overwritten`);\n } else {\n systemEnv[key] = parsed.env[key];\n loadedEnvKeys.push(key);\n }\n }\n\n // Mark the environment as loaded\n systemEnv[LOADED_ENV_NAME] = JSON.stringify(loadedEnvKeys);\n\n return { result: 'loaded' as const, ...parsed, loaded: loadedEnvKeys };\n}\n\n/**\n * Parse all environment variables using the detected list of `.env*` files from a project.\n * This does not check for collisions of existing system environment variables, or mutates the system environment variables.\n */\nexport function parseProjectEnv(\n projectRoot: string,\n options?: Parameters<typeof getEnvFiles>[0] & Parameters<typeof parseEnvFiles>[1]\n) {\n return parseEnvFiles(\n getEnvFiles(options).map((envFile) => path.join(projectRoot, envFile)),\n options\n );\n}\n\n/**\n * Parse all environment variables using the detected list of `.env*` files from a project.\n * This won't override existing environment variables defined in the system environment.\n * Once the mutations are done, this will also set a propert `__EXPO_ENV=true` on the system env to avoid multiple mutations.\n * This check can be disabled through `{ force: true }`.\n */\nexport function loadProjectEnv(\n projectRoot: string,\n options?: Parameters<typeof getEnvFiles>[0] & Parameters<typeof loadEnvFiles>[1]\n) {\n return loadEnvFiles(\n getEnvFiles(options).map((envFile) => path.join(projectRoot, envFile)),\n options\n );\n}\n\n/** Log the loaded environment info from the loaded results */\nexport function logLoadedEnv(\n envInfo: ReturnType<typeof loadEnvFiles>,\n options: Parameters<typeof loadEnvFiles>[1] = {}\n) {\n // Skip when running in force mode, or no environment variables are loaded\n if (options.force || options.silent || !envInfo.loaded.length) return envInfo;\n\n // Log the loaded environment files, when not skipped\n if (envInfo.result === 'loaded') {\n console.log(\n chalk.gray('env: load', envInfo.files.map((file) => path.basename(file)).join(' '))\n );\n }\n\n // Log the loaded environment variables\n console.log(chalk.gray('env: export', envInfo.loaded.join(' ')));\n\n return envInfo;\n}\n\n// Legacy API - for backwards compatibility\n\nlet memo: ReturnType<typeof parseEnvFiles> | null = null;\n\n/**\n * Get the environment variables without mutating the environment.\n * This returns memoized values unless the `force` property is provided.\n *\n * @deprecated use {@link parseProjectEnv} instead\n */\nexport function get(\n projectRoot: string,\n {\n force,\n silent,\n }: {\n force?: boolean;\n silent?: boolean;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return { env: {}, files: [] };\n }\n if (force || !memo) {\n memo = parseProjectEnv(projectRoot, { silent });\n }\n return memo;\n}\n\n/**\n * Load environment variables from .env files and mutate the current `process.env` with the results.\n *\n * @deprecated use {@link loadProjectEnv} instead\n */\nexport function load(\n projectRoot: string,\n options: {\n force?: boolean;\n silent?: boolean;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return process.env;\n }\n\n const envInfo = get(projectRoot, options);\n const loadedEnvKeys: string[] = [];\n\n for (const key in envInfo.env) {\n if (typeof process.env[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and IS NOT overwritten`);\n } else {\n // Avoid creating a new object, mutate it instead as this causes problems in Bun\n process.env[key] = envInfo.env[key];\n loadedEnvKeys.push(key);\n }\n }\n\n // Port the result of `get` to the newer result object\n logLoadedEnv({ ...envInfo, result: 'loaded', loaded: loadedEnvKeys }, options);\n\n return process.env;\n}\n\n/**\n * Get a list of all `.env*` files based on the `NODE_ENV` mode.\n * This returns a list of files, in order of highest priority to lowest priority.\n *\n * @deprecated use {@link getEnvFiles} instead\n * @see https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n */\nexport function getFiles(mode: string | undefined, { silent = false }: { silent?: boolean } = {}) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return [];\n }\n\n return getEnvFiles({ mode, silent });\n}\n\n/**\n * Parses the contents of a single `.env` file, optionally expanding it immediately.\n */\nexport function parseEnv(contents: string, sourceEnv?: EnvOutput): EnvOutput {\n try {\n const env = parse(contents);\n return expand(env, sourceEnv || {});\n } catch {\n return {};\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"index.js","names":["_chalk","data","_interopRequireDefault","require","_getenv","_nodeConsole","_nodeFs","_nodePath","_constants","_parse","e","__esModule","default","debug","ORIGINAL_ENV_BACKUP_KEY","Symbol","for","globalStore","globalThis","originalEnvBackup","WeakMap","rememberOriginal","systemEnv","key","isUnsafeAllowedEnvKey","backup","get","Map","set","has","isEnabled","boolish","KNOWN_MODES","exports","LOADED_ENV_NAME","getEnvFiles","mode","process","env","NODE_ENV","silent","logError","console","error","logWarning","warn","includes","filter","Boolean","parseEnvFiles","envFiles","files","sensitiveLoadedKeys","loadedEnvVars","loadedEnvFiles","blockedByFile","localOnlyByFile","sensitive","Set","reverse","forEach","envFile","envFileContent","fs","readFileSync","envFileParsed","parse","isLocalFile","path","basename","endsWith","push","Object","keys","isIgnoredEnvKey","isLocalEnvKey","add","code","violations","length","formatBlockedViolation","formatLocalOnlyViolation","Error","join","expand","formatViolationFiles","byFile","entries","map","file","loadEnvFiles","force","result","loaded","JSON","parsed","loadedEnvKeys","stringify","parseProjectEnv","projectRoot","options","loadProjectEnv","getOriginalEnv","original","undefined","getOriginalEnvValue","logLoadedEnv","envInfo","log","chalk","gray","yellow","memo","load","getFiles","parseEnv","contents","sourceEnv"],"sources":["../src/index.ts"],"sourcesContent":["import chalk from 'chalk';\nimport { boolish } from 'getenv';\nimport console from 'node:console';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\nimport { isIgnoredEnvKey, isLocalEnvKey, isUnsafeAllowedEnvKey } from './constants';\nimport { parse, expand, type EnvOutput } from './parse';\n\nconst debug = require('debug')('expo:env') as typeof console.log;\n\ntype EnvBackupStore = WeakMap<EnvOutput, Map<string, string | undefined>>;\n\nconst ORIGINAL_ENV_BACKUP_KEY = Symbol.for('@expo/env.originalEnvBackup.v1');\nconst globalStore = globalThis as Record<symbol, EnvBackupStore | undefined>;\nconst originalEnvBackup: EnvBackupStore =\n globalStore[ORIGINAL_ENV_BACKUP_KEY] ?? (globalStore[ORIGINAL_ENV_BACKUP_KEY] = new WeakMap());\n\nfunction rememberOriginal(systemEnv: EnvOutput, key: string): void {\n if (isUnsafeAllowedEnvKey(key)) return;\n let backup = originalEnvBackup.get(systemEnv);\n if (!backup) {\n backup = new Map();\n originalEnvBackup.set(systemEnv, backup);\n }\n if (!backup.has(key)) {\n backup.set(key, systemEnv[key]);\n }\n}\n\n/** Determine if the `.env` files are enabled or not, through `EXPO_NO_DOTENV` */\nexport function isEnabled() {\n return !boolish('EXPO_NO_DOTENV', false);\n}\n\n/** All conventional modes that should not cause warnings */\nexport const KNOWN_MODES = ['development', 'test', 'production'];\n\n/** The environment variable name to use when marking the environment as loaded */\nexport const LOADED_ENV_NAME = '__EXPO_ENV_LOADED';\n\n/**\n * Get a list of all `.env*` files based on the `NODE_ENV` mode.\n * This returns a list of files, in order of highest priority to lowest priority.\n *\n * @see https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n */\nexport function getEnvFiles({\n mode = process.env.NODE_ENV,\n silent,\n}: {\n /** The mode to use when creating the list of `.env*` files, defaults to `NODE_ENV` */\n mode?: string;\n /** If possible misconfiguration warnings should be logged, or only logged as debug log */\n silent?: boolean;\n} = {}) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return [];\n }\n\n const logError = silent ? debug : console.error;\n const logWarning = silent ? debug : console.warn;\n\n if (!mode) {\n logError(\n `The NODE_ENV environment variable is required but was not specified. Ensure the project is bundled with Expo CLI or NODE_ENV is set. Using only .env.local and .env`\n );\n return ['.env.local', '.env'];\n }\n\n if (!KNOWN_MODES.includes(mode)) {\n logWarning(\n `NODE_ENV=\"${mode}\" is non-conventional and might cause development code to run in production. Use \"development\", \"test\", or \"production\" instead. Continuing with non-conventional mode`\n );\n }\n\n // see: https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n return [\n `.env.${mode}.local`,\n // Don't include `.env.local` for `test` environment\n // since normally you expect tests to produce the same\n // results for everyone\n mode !== 'test' && `.env.local`,\n `.env.${mode}`,\n `.env`,\n ].filter(Boolean) as string[];\n}\n\n/**\n * Parse all environment variables using the list of `.env*` files, in order of higest priority to lowest priority.\n * This does not check for collisions of existing system environment variables, or mutates the system environment variables.\n */\nexport function parseEnvFiles(\n envFiles: string[],\n {\n systemEnv = process.env,\n }: {\n /** The system environment to use when expanding environment variables, defaults to `process.env` */\n systemEnv?: EnvOutput;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return { env: {}, files: [], sensitiveLoadedKeys: [] as string[] };\n }\n\n // Load environment variables from .env* files. Suppress warnings using silent\n // if this file is missing. Dotenv will only parse the environment variables,\n // `@expo/env` will set the resulting variables to the current process.\n // Variable expansion is supported in .env files, and executed as final step.\n // https://github.com/motdotla/dotenv\n // https://github.com/motdotla/dotenv-expand\n const loadedEnvVars: EnvOutput = {};\n const loadedEnvFiles: string[] = [];\n const blockedByFile: Record<string, string[]> = {};\n const localOnlyByFile: Record<string, string[]> = {};\n const sensitive = new Set<string>();\n\n // Iterate over each dotenv file in lowest prio to highest prio order.\n // This step won't write to the process.env, but will overwrite the parsed envs.\n [...envFiles].reverse().forEach((envFile) => {\n try {\n const envFileContent = fs.readFileSync(envFile, 'utf8');\n const envFileParsed = parse(envFileContent);\n const isLocalFile = path.basename(envFile).endsWith('.local');\n\n loadedEnvFiles.push(envFile);\n debug(`Loaded environment variables from: ${envFile}`);\n\n for (const key of Object.keys(envFileParsed)) {\n if (isIgnoredEnvKey(key)) {\n (blockedByFile[envFile] ||= []).push(key);\n debug(`\"${key}\" is blocked from dotenv files, skipping in: ${envFile}`);\n continue;\n }\n if (!isLocalFile && isLocalEnvKey(key)) {\n (localOnlyByFile[envFile] ||= []).push(key);\n debug(`\"${key}\" is only allowed in .local env files, skipping in: ${envFile}`);\n continue;\n }\n if (isLocalFile && isLocalEnvKey(key)) {\n sensitive.add(key);\n }\n if (typeof loadedEnvVars[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and overwritten by: ${envFile}`);\n }\n\n loadedEnvVars[key] = envFileParsed[key];\n }\n } catch (error: any) {\n if ('code' in error && error.code === 'ENOENT') {\n return debug(`${envFile} does not exist, skipping this env file`);\n }\n if ('code' in error && error.code === 'EISDIR') {\n return debug(`${envFile} is a directory, skipping this env file`);\n }\n if ('code' in error && error.code === 'EACCES') {\n return debug(`No permission to read ${envFile}, skipping this env file`);\n }\n\n throw error;\n }\n });\n\n const violations: string[] = [];\n if (Object.keys(blockedByFile).length > 0) {\n violations.push(formatBlockedViolation(blockedByFile));\n }\n if (Object.keys(localOnlyByFile).length > 0) {\n violations.push(formatLocalOnlyViolation(localOnlyByFile));\n }\n if (violations.length > 0) {\n throw new Error(violations.join('\\n\\n'));\n }\n\n const env = expand(loadedEnvVars, systemEnv);\n for (const key in env) {\n rememberOriginal(systemEnv, key);\n }\n\n return {\n env,\n files: loadedEnvFiles.reverse(),\n sensitiveLoadedKeys: [...sensitive],\n };\n}\n\nfunction formatViolationFiles(byFile: Record<string, string[]>): string {\n return Object.entries(byFile)\n .map(([file, keys]) => ` ${path.basename(file)}: ${keys.join(', ')}`)\n .join('\\n');\n}\n\nfunction formatBlockedViolation(byFile: Record<string, string[]>): string {\n return [\n 'Refused to load dangerous environment variables from .env files.',\n 'Opt in via EXPO_UNSAFE_DOTENV_KEYS in your shell environment if you truly need them.',\n '',\n formatViolationFiles(byFile),\n ].join('\\n');\n}\n\nfunction formatLocalOnlyViolation(byFile: Record<string, string[]>): string {\n return [\n 'Refused to load personal environment variables from a non-.local env file.',\n 'Move them to a .local env file.',\n '',\n formatViolationFiles(byFile),\n ].join('\\n');\n}\n\n/**\n * Parse all environment variables using the list of `.env*` files, and mutate the system environment with these variables.\n * This won't override existing environment variables defined in the system environment.\n * Once the mutations are done, this will also set a propert `__EXPO_ENV=true` on the system env to avoid multiple mutations.\n * This check can be disabled through `{ force: true }`.\n */\nexport function loadEnvFiles(\n envFiles: string[],\n {\n force,\n silent = false,\n systemEnv = process.env,\n }: Parameters<typeof parseEnvFiles>[1] & {\n /** If the environment variables should be applied to the system environment, regardless of previous mutations */\n force?: boolean;\n /** If possible misconfiguration warnings should be logged, or only logged as debug log */\n silent?: boolean;\n } = {}\n) {\n if (!force && systemEnv[LOADED_ENV_NAME]) {\n return { result: 'skipped' as const, loaded: JSON.parse(systemEnv[LOADED_ENV_NAME]) };\n }\n\n const parsed = parseEnvFiles(envFiles, { systemEnv });\n const loadedEnvKeys: string[] = [];\n\n for (const key in parsed.env) {\n if (typeof systemEnv[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and IS NOT overwritten`);\n } else {\n rememberOriginal(systemEnv, key);\n systemEnv[key] = parsed.env[key];\n loadedEnvKeys.push(key);\n }\n }\n\n // Mark the environment as loaded\n rememberOriginal(systemEnv, LOADED_ENV_NAME);\n systemEnv[LOADED_ENV_NAME] = JSON.stringify(loadedEnvKeys);\n\n return { result: 'loaded' as const, ...parsed, loaded: loadedEnvKeys };\n}\n\n/**\n * Parse all environment variables using the detected list of `.env*` files from a project.\n * This does not check for collisions of existing system environment variables, or mutates the system environment variables.\n */\nexport function parseProjectEnv(\n projectRoot: string,\n options?: Parameters<typeof getEnvFiles>[0] & Parameters<typeof parseEnvFiles>[1]\n) {\n return parseEnvFiles(\n getEnvFiles(options).map((envFile) => path.join(projectRoot, envFile)),\n options\n );\n}\n\n/**\n * Parse all environment variables using the detected list of `.env*` files from a project.\n * This won't override existing environment variables defined in the system environment.\n * Once the mutations are done, this will also set a propert `__EXPO_ENV=true` on the system env to avoid multiple mutations.\n * This check can be disabled through `{ force: true }`.\n */\nexport function loadProjectEnv(\n projectRoot: string,\n options?: Parameters<typeof getEnvFiles>[0] & Parameters<typeof loadEnvFiles>[1]\n) {\n return loadEnvFiles(\n getEnvFiles(options).map((envFile) => path.join(projectRoot, envFile)),\n options\n );\n}\n\n/**\n * Get a fresh clone of the system environment with all `@expo/env`-applied\n * mutations reverted to their pre-load values. The result is intended to be\n * passed as the `env` option of `child_process.spawn` / `@expo/spawn-async`\n * when a subprocess should observe the environment as it was before any\n * `.env*` files were loaded — for example, when resolving SDK tooling paths\n * that should not be influenced by project-controlled `.env` values.\n *\n * Allocates lazily: nothing is held until this function is called, and each\n * call returns a new object so callers may mutate it freely.\n *\n * @param systemEnv The env to revert against; defaults to `process.env`.\n */\nexport function getOriginalEnv(systemEnv: EnvOutput = process.env): EnvOutput {\n const result: EnvOutput = { ...systemEnv };\n const backup = originalEnvBackup.get(systemEnv);\n if (backup) {\n for (const [key, original] of backup) {\n if (original === undefined) {\n delete result[key];\n } else {\n result[key] = original;\n }\n }\n }\n return result;\n}\n\n/**\n * Get the pre-load value of a single environment variable as recorded by\n * `@expo/env`. Falls through to the value in `systemEnv` for keys that\n * `@expo/env` never touched. O(1) and allocation-free, intended for read-sites\n * that resolve filesystem paths or executables from a single env var.\n *\n * Honors `EXPO_UNSAFE_DOTENV_KEYS`: keys the caller has explicitly opted into\n * via the escape hatch return their currently loaded value, not the original.\n *\n * @param key The environment variable to read.\n * @param systemEnv The env to read against; defaults to `process.env`.\n */\nexport function getOriginalEnvValue(\n key: string,\n systemEnv: EnvOutput = process.env\n): string | undefined {\n const backup = originalEnvBackup.get(systemEnv);\n if (backup && backup.has(key)) {\n return backup.get(key);\n }\n return systemEnv[key];\n}\n\n/** Log the loaded environment info from the loaded results */\nexport function logLoadedEnv(\n envInfo: ReturnType<typeof loadEnvFiles>,\n options: Parameters<typeof loadEnvFiles>[1] = {}\n) {\n // Skip when running in force mode, or no environment variables are loaded\n if (options.force || options.silent || !envInfo.loaded.length) return envInfo;\n\n // Log the loaded environment files, when not skipped\n if (envInfo.result === 'loaded') {\n console.log(\n chalk.gray('env: load', envInfo.files.map((file) => path.basename(file)).join(' '))\n );\n }\n\n // Log the loaded environment variables\n console.log(chalk.gray('env: export', envInfo.loaded.join(' ')));\n\n // Highlight developer-tool roots / secrets that were loaded from a .local file —\n // the same keys would be refused from any non-.local file. Surfacing them here\n // tells the user which \"sensitive\" values are influencing the build.\n if (envInfo.result === 'loaded' && envInfo.sensitiveLoadedKeys?.length) {\n console.log(chalk.yellow('env: export (sensitive)', envInfo.sensitiveLoadedKeys.join(' ')));\n }\n\n return envInfo;\n}\n\n// Legacy API - for backwards compatibility\n\nlet memo: ReturnType<typeof parseEnvFiles> | null = null;\n\n/**\n * Get the environment variables without mutating the environment.\n * This returns memoized values unless the `force` property is provided.\n *\n * @deprecated use {@link parseProjectEnv} instead\n */\nexport function get(\n projectRoot: string,\n {\n force,\n silent,\n }: {\n force?: boolean;\n silent?: boolean;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return { env: {}, files: [], sensitiveLoadedKeys: [] as string[] };\n }\n if (force || !memo) {\n memo = parseProjectEnv(projectRoot, { silent });\n }\n return memo;\n}\n\n/**\n * Load environment variables from .env files and mutate the current `process.env` with the results.\n *\n * @deprecated use {@link loadProjectEnv} instead\n */\nexport function load(\n projectRoot: string,\n options: {\n force?: boolean;\n silent?: boolean;\n } = {}\n) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return process.env;\n }\n\n const envInfo = get(projectRoot, options);\n const loadedEnvKeys: string[] = [];\n\n for (const key in envInfo.env) {\n if (typeof process.env[key] !== 'undefined') {\n debug(`\"${key}\" is already defined and IS NOT overwritten`);\n } else {\n // Avoid creating a new object, mutate it instead as this causes problems in Bun\n rememberOriginal(process.env, key);\n process.env[key] = envInfo.env[key];\n loadedEnvKeys.push(key);\n }\n }\n\n // Port the result of `get` to the newer result object\n logLoadedEnv({ ...envInfo, result: 'loaded', loaded: loadedEnvKeys }, options);\n\n return process.env;\n}\n\n/**\n * Get a list of all `.env*` files based on the `NODE_ENV` mode.\n * This returns a list of files, in order of highest priority to lowest priority.\n *\n * @deprecated use {@link getEnvFiles} instead\n * @see https://github.com/bkeepers/dotenv/tree/v3.1.4#customizing-rails\n */\nexport function getFiles(mode: string | undefined, { silent = false }: { silent?: boolean } = {}) {\n if (!isEnabled()) {\n debug(`Skipping .env files because EXPO_NO_DOTENV is defined`);\n return [];\n }\n\n return getEnvFiles({ mode, silent });\n}\n\n/**\n * Parses the contents of a single `.env` file, optionally expanding it immediately.\n */\nexport function parseEnv(contents: string, sourceEnv?: EnvOutput): EnvOutput {\n try {\n const env = parse(contents);\n for (const key in env) {\n if (isIgnoredEnvKey(key)) {\n delete env[key];\n }\n }\n return expand(env, sourceEnv || {});\n } catch {\n return {};\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;AAAA,SAAAA,OAAA;EAAA,MAAAC,IAAA,GAAAC,sBAAA,CAAAC,OAAA;EAAAH,MAAA,YAAAA,CAAA;IAAA,OAAAC,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AACA,SAAAG,QAAA;EAAA,MAAAH,IAAA,GAAAE,OAAA;EAAAC,OAAA,YAAAA,CAAA;IAAA,OAAAH,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AACA,SAAAI,aAAA;EAAA,MAAAJ,IAAA,GAAAC,sBAAA,CAAAC,OAAA;EAAAE,YAAA,YAAAA,CAAA;IAAA,OAAAJ,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AACA,SAAAK,QAAA;EAAA,MAAAL,IAAA,GAAAC,sBAAA,CAAAC,OAAA;EAAAG,OAAA,YAAAA,CAAA;IAAA,OAAAL,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AACA,SAAAM,UAAA;EAAA,MAAAN,IAAA,GAAAC,sBAAA,CAAAC,OAAA;EAAAI,SAAA,YAAAA,CAAA;IAAA,OAAAN,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AAEA,SAAAO,WAAA;EAAA,MAAAP,IAAA,GAAAE,OAAA;EAAAK,UAAA,YAAAA,CAAA;IAAA,OAAAP,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AACA,SAAAQ,OAAA;EAAA,MAAAR,IAAA,GAAAE,OAAA;EAAAM,MAAA,YAAAA,CAAA;IAAA,OAAAR,IAAA;EAAA;EAAA,OAAAA,IAAA;AAAA;AAAwD,SAAAC,uBAAAQ,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAExD,MAAMG,KAAK,GAAGV,OAAO,CAAC,OAAO,CAAC,CAAC,UAAU,CAAuB;AAIhE,MAAMW,uBAAuB,GAAGC,MAAM,CAACC,GAAG,CAAC,gCAAgC,CAAC;AAC5E,MAAMC,WAAW,GAAGC,UAAwD;AAC5E,MAAMC,iBAAiC,GACrCF,WAAW,CAACH,uBAAuB,CAAC,KAAKG,WAAW,CAACH,uBAAuB,CAAC,GAAG,IAAIM,OAAO,CAAC,CAAC,CAAC;AAEhG,SAASC,gBAAgBA,CAACC,SAAoB,EAAEC,GAAW,EAAQ;EACjE,IAAI,IAAAC,kCAAqB,EAACD,GAAG,CAAC,EAAE;EAChC,IAAIE,MAAM,GAAGN,iBAAiB,CAACO,GAAG,CAACJ,SAAS,CAAC;EAC7C,IAAI,CAACG,MAAM,EAAE;IACXA,MAAM,GAAG,IAAIE,GAAG,CAAC,CAAC;IAClBR,iBAAiB,CAACS,GAAG,CAACN,SAAS,EAAEG,MAAM,CAAC;EAC1C;EACA,IAAI,CAACA,MAAM,CAACI,GAAG,CAACN,GAAG,CAAC,EAAE;IACpBE,MAAM,CAACG,GAAG,CAACL,GAAG,EAAED,SAAS,CAACC,GAAG,CAAC,CAAC;EACjC;AACF;;AAEA;AACO,SAASO,SAASA,CAAA,EAAG;EAC1B,OAAO,CAAC,IAAAC,iBAAO,EAAC,gBAAgB,EAAE,KAAK,CAAC;AAC1C;;AAEA;AACO,MAAMC,WAAW,GAAAC,OAAA,CAAAD,WAAA,GAAG,CAAC,aAAa,EAAE,MAAM,EAAE,YAAY,CAAC;;AAEhE;AACO,MAAME,eAAe,GAAAD,OAAA,CAAAC,eAAA,GAAG,mBAAmB;;AAElD;AACA;AACA;AACA;AACA;AACA;AACO,SAASC,WAAWA,CAAC;EAC1BC,IAAI,GAAGC,OAAO,CAACC,GAAG,CAACC,QAAQ;EAC3BC;AAMF,CAAC,GAAG,CAAC,CAAC,EAAE;EACN,IAAI,CAACV,SAAS,CAAC,CAAC,EAAE;IAChBjB,KAAK,CAAC,uDAAuD,CAAC;IAC9D,OAAO,EAAE;EACX;EAEA,MAAM4B,QAAQ,GAAGD,MAAM,GAAG3B,KAAK,GAAG6B,sBAAO,CAACC,KAAK;EAC/C,MAAMC,UAAU,GAAGJ,MAAM,GAAG3B,KAAK,GAAG6B,sBAAO,CAACG,IAAI;EAEhD,IAAI,CAACT,IAAI,EAAE;IACTK,QAAQ,CACN,qKACF,CAAC;IACD,OAAO,CAAC,YAAY,EAAE,MAAM,CAAC;EAC/B;EAEA,IAAI,CAACT,WAAW,CAACc,QAAQ,CAACV,IAAI,CAAC,EAAE;IAC/BQ,UAAU,CACR,aAAaR,IAAI,wKACnB,CAAC;EACH;;EAEA;EACA,OAAO,CACL,QAAQA,IAAI,QAAQ;EACpB;EACA;EACA;EACAA,IAAI,KAAK,MAAM,IAAI,YAAY,EAC/B,QAAQA,IAAI,EAAE,EACd,MAAM,CACP,CAACW,MAAM,CAACC,OAAO,CAAC;AACnB;;AAEA;AACA;AACA;AACA;AACO,SAASC,aAAaA,CAC3BC,QAAkB,EAClB;EACE5B,SAAS,GAAGe,OAAO,CAACC;AAItB,CAAC,GAAG,CAAC,CAAC,EACN;EACA,IAAI,CAACR,SAAS,CAAC,CAAC,EAAE;IAChBjB,KAAK,CAAC,uDAAuD,CAAC;IAC9D,OAAO;MAAEyB,GAAG,EAAE,CAAC,CAAC;MAAEa,KAAK,EAAE,EAAE;MAAEC,mBAAmB,EAAE;IAAe,CAAC;EACpE;;EAEA;EACA;EACA;EACA;EACA;EACA;EACA,MAAMC,aAAwB,GAAG,CAAC,CAAC;EACnC,MAAMC,cAAwB,GAAG,EAAE;EACnC,MAAMC,aAAuC,GAAG,CAAC,CAAC;EAClD,MAAMC,eAAyC,GAAG,CAAC,CAAC;EACpD,MAAMC,SAAS,GAAG,IAAIC,GAAG,CAAS,CAAC;;EAEnC;EACA;EACA,CAAC,GAAGR,QAAQ,CAAC,CAACS,OAAO,CAAC,CAAC,CAACC,OAAO,CAAEC,OAAO,IAAK;IAC3C,IAAI;MACF,MAAMC,cAAc,GAAGC,iBAAE,CAACC,YAAY,CAACH,OAAO,EAAE,MAAM,CAAC;MACvD,MAAMI,aAAa,GAAG,IAAAC,cAAK,EAACJ,cAAc,CAAC;MAC3C,MAAMK,WAAW,GAAGC,mBAAI,CAACC,QAAQ,CAACR,OAAO,CAAC,CAACS,QAAQ,CAAC,QAAQ,CAAC;MAE7DhB,cAAc,CAACiB,IAAI,CAACV,OAAO,CAAC;MAC5BhD,KAAK,CAAC,sCAAsCgD,OAAO,EAAE,CAAC;MAEtD,KAAK,MAAMtC,GAAG,IAAIiD,MAAM,CAACC,IAAI,CAACR,aAAa,CAAC,EAAE;QAC5C,IAAI,IAAAS,4BAAe,EAACnD,GAAG,CAAC,EAAE;UACxB,CAACgC,aAAa,CAACM,OAAO,CAAC,KAAK,EAAE,EAAEU,IAAI,CAAChD,GAAG,CAAC;UACzCV,KAAK,CAAC,IAAIU,GAAG,gDAAgDsC,OAAO,EAAE,CAAC;UACvE;QACF;QACA,IAAI,CAACM,WAAW,IAAI,IAAAQ,0BAAa,EAACpD,GAAG,CAAC,EAAE;UACtC,CAACiC,eAAe,CAACK,OAAO,CAAC,KAAK,EAAE,EAAEU,IAAI,CAAChD,GAAG,CAAC;UAC3CV,KAAK,CAAC,IAAIU,GAAG,uDAAuDsC,OAAO,EAAE,CAAC;UAC9E;QACF;QACA,IAAIM,WAAW,IAAI,IAAAQ,0BAAa,EAACpD,GAAG,CAAC,EAAE;UACrCkC,SAAS,CAACmB,GAAG,CAACrD,GAAG,CAAC;QACpB;QACA,IAAI,OAAO8B,aAAa,CAAC9B,GAAG,CAAC,KAAK,WAAW,EAAE;UAC7CV,KAAK,CAAC,IAAIU,GAAG,4CAA4CsC,OAAO,EAAE,CAAC;QACrE;QAEAR,aAAa,CAAC9B,GAAG,CAAC,GAAG0C,aAAa,CAAC1C,GAAG,CAAC;MACzC;IACF,CAAC,CAAC,OAAOoB,KAAU,EAAE;MACnB,IAAI,MAAM,IAAIA,KAAK,IAAIA,KAAK,CAACkC,IAAI,KAAK,QAAQ,EAAE;QAC9C,OAAOhE,KAAK,CAAC,GAAGgD,OAAO,yCAAyC,CAAC;MACnE;MACA,IAAI,MAAM,IAAIlB,KAAK,IAAIA,KAAK,CAACkC,IAAI,KAAK,QAAQ,EAAE;QAC9C,OAAOhE,KAAK,CAAC,GAAGgD,OAAO,yCAAyC,CAAC;MACnE;MACA,IAAI,MAAM,IAAIlB,KAAK,IAAIA,KAAK,CAACkC,IAAI,KAAK,QAAQ,EAAE;QAC9C,OAAOhE,KAAK,CAAC,yBAAyBgD,OAAO,0BAA0B,CAAC;MAC1E;MAEA,MAAMlB,KAAK;IACb;EACF,CAAC,CAAC;EAEF,MAAMmC,UAAoB,GAAG,EAAE;EAC/B,IAAIN,MAAM,CAACC,IAAI,CAAClB,aAAa,CAAC,CAACwB,MAAM,GAAG,CAAC,EAAE;IACzCD,UAAU,CAACP,IAAI,CAACS,sBAAsB,CAACzB,aAAa,CAAC,CAAC;EACxD;EACA,IAAIiB,MAAM,CAACC,IAAI,CAACjB,eAAe,CAAC,CAACuB,MAAM,GAAG,CAAC,EAAE;IAC3CD,UAAU,CAACP,IAAI,CAACU,wBAAwB,CAACzB,eAAe,CAAC,CAAC;EAC5D;EACA,IAAIsB,UAAU,CAACC,MAAM,GAAG,CAAC,EAAE;IACzB,MAAM,IAAIG,KAAK,CAACJ,UAAU,CAACK,IAAI,CAAC,MAAM,CAAC,CAAC;EAC1C;EAEA,MAAM7C,GAAG,GAAG,IAAA8C,eAAM,EAAC/B,aAAa,EAAE/B,SAAS,CAAC;EAC5C,KAAK,MAAMC,GAAG,IAAIe,GAAG,EAAE;IACrBjB,gBAAgB,CAACC,SAAS,EAAEC,GAAG,CAAC;EAClC;EAEA,OAAO;IACLe,GAAG;IACHa,KAAK,EAAEG,cAAc,CAACK,OAAO,CAAC,CAAC;IAC/BP,mBAAmB,EAAE,CAAC,GAAGK,SAAS;EACpC,CAAC;AACH;AAEA,SAAS4B,oBAAoBA,CAACC,MAAgC,EAAU;EACtE,OAAOd,MAAM,CAACe,OAAO,CAACD,MAAM,CAAC,CAC1BE,GAAG,CAAC,CAAC,CAACC,IAAI,EAAEhB,IAAI,CAAC,KAAK,KAAKL,mBAAI,CAACC,QAAQ,CAACoB,IAAI,CAAC,KAAKhB,IAAI,CAACU,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CACrEA,IAAI,CAAC,IAAI,CAAC;AACf;AAEA,SAASH,sBAAsBA,CAACM,MAAgC,EAAU;EACxE,OAAO,CACL,kEAAkE,EAClE,sFAAsF,EACtF,EAAE,EACFD,oBAAoB,CAACC,MAAM,CAAC,CAC7B,CAACH,IAAI,CAAC,IAAI,CAAC;AACd;AAEA,SAASF,wBAAwBA,CAACK,MAAgC,EAAU;EAC1E,OAAO,CACL,4EAA4E,EAC5E,iCAAiC,EACjC,EAAE,EACFD,oBAAoB,CAACC,MAAM,CAAC,CAC7B,CAACH,IAAI,CAAC,IAAI,CAAC;AACd;;AAEA;AACA;AACA;AACA;AACA;AACA;AACO,SAASO,YAAYA,CAC1BxC,QAAkB,EAClB;EACEyC,KAAK;EACLnD,MAAM,GAAG,KAAK;EACdlB,SAAS,GAAGe,OAAO,CAACC;AAMtB,CAAC,GAAG,CAAC,CAAC,EACN;EACA,IAAI,CAACqD,KAAK,IAAIrE,SAAS,CAACY,eAAe,CAAC,EAAE;IACxC,OAAO;MAAE0D,MAAM,EAAE,SAAkB;MAAEC,MAAM,EAAEC,IAAI,CAAC5B,KAAK,CAAC5C,SAAS,CAACY,eAAe,CAAC;IAAE,CAAC;EACvF;EAEA,MAAM6D,MAAM,GAAG9C,aAAa,CAACC,QAAQ,EAAE;IAAE5B;EAAU,CAAC,CAAC;EACrD,MAAM0E,aAAuB,GAAG,EAAE;EAElC,KAAK,MAAMzE,GAAG,IAAIwE,MAAM,CAACzD,GAAG,EAAE;IAC5B,IAAI,OAAOhB,SAAS,CAACC,GAAG,CAAC,KAAK,WAAW,EAAE;MACzCV,KAAK,CAAC,IAAIU,GAAG,6CAA6C,CAAC;IAC7D,CAAC,MAAM;MACLF,gBAAgB,CAACC,SAAS,EAAEC,GAAG,CAAC;MAChCD,SAAS,CAACC,GAAG,CAAC,GAAGwE,MAAM,CAACzD,GAAG,CAACf,GAAG,CAAC;MAChCyE,aAAa,CAACzB,IAAI,CAAChD,GAAG,CAAC;IACzB;EACF;;EAEA;EACAF,gBAAgB,CAACC,SAAS,EAAEY,eAAe,CAAC;EAC5CZ,SAAS,CAACY,eAAe,CAAC,GAAG4D,IAAI,CAACG,SAAS,CAACD,aAAa,CAAC;EAE1D,OAAO;IAAEJ,MAAM,EAAE,QAAiB;IAAE,GAAGG,MAAM;IAAEF,MAAM,EAAEG;EAAc,CAAC;AACxE;;AAEA;AACA;AACA;AACA;AACO,SAASE,eAAeA,CAC7BC,WAAmB,EACnBC,OAAiF,EACjF;EACA,OAAOnD,aAAa,CAClBd,WAAW,CAACiE,OAAO,CAAC,CAACZ,GAAG,CAAE3B,OAAO,IAAKO,mBAAI,CAACe,IAAI,CAACgB,WAAW,EAAEtC,OAAO,CAAC,CAAC,EACtEuC,OACF,CAAC;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACO,SAASC,cAAcA,CAC5BF,WAAmB,EACnBC,OAAgF,EAChF;EACA,OAAOV,YAAY,CACjBvD,WAAW,CAACiE,OAAO,CAAC,CAACZ,GAAG,CAAE3B,OAAO,IAAKO,mBAAI,CAACe,IAAI,CAACgB,WAAW,EAAEtC,OAAO,CAAC,CAAC,EACtEuC,OACF,CAAC;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAASE,cAAcA,CAAChF,SAAoB,GAAGe,OAAO,CAACC,GAAG,EAAa;EAC5E,MAAMsD,MAAiB,GAAG;IAAE,GAAGtE;EAAU,CAAC;EAC1C,MAAMG,MAAM,GAAGN,iBAAiB,CAACO,GAAG,CAACJ,SAAS,CAAC;EAC/C,IAAIG,MAAM,EAAE;IACV,KAAK,MAAM,CAACF,GAAG,EAAEgF,QAAQ,CAAC,IAAI9E,MAAM,EAAE;MACpC,IAAI8E,QAAQ,KAAKC,SAAS,EAAE;QAC1B,OAAOZ,MAAM,CAACrE,GAAG,CAAC;MACpB,CAAC,MAAM;QACLqE,MAAM,CAACrE,GAAG,CAAC,GAAGgF,QAAQ;MACxB;IACF;EACF;EACA,OAAOX,MAAM;AACf;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAASa,mBAAmBA,CACjClF,GAAW,EACXD,SAAoB,GAAGe,OAAO,CAACC,GAAG,EACd;EACpB,MAAMb,MAAM,GAAGN,iBAAiB,CAACO,GAAG,CAACJ,SAAS,CAAC;EAC/C,IAAIG,MAAM,IAAIA,MAAM,CAACI,GAAG,CAACN,GAAG,CAAC,EAAE;IAC7B,OAAOE,MAAM,CAACC,GAAG,CAACH,GAAG,CAAC;EACxB;EACA,OAAOD,SAAS,CAACC,GAAG,CAAC;AACvB;;AAEA;AACO,SAASmF,YAAYA,CAC1BC,OAAwC,EACxCP,OAA2C,GAAG,CAAC,CAAC,EAChD;EACA;EACA,IAAIA,OAAO,CAACT,KAAK,IAAIS,OAAO,CAAC5D,MAAM,IAAI,CAACmE,OAAO,CAACd,MAAM,CAACd,MAAM,EAAE,OAAO4B,OAAO;;EAE7E;EACA,IAAIA,OAAO,CAACf,MAAM,KAAK,QAAQ,EAAE;IAC/BlD,sBAAO,CAACkE,GAAG,CACTC,gBAAK,CAACC,IAAI,CAAC,WAAW,EAAEH,OAAO,CAACxD,KAAK,CAACqC,GAAG,CAAEC,IAAI,IAAKrB,mBAAI,CAACC,QAAQ,CAACoB,IAAI,CAAC,CAAC,CAACN,IAAI,CAAC,GAAG,CAAC,CACpF,CAAC;EACH;;EAEA;EACAzC,sBAAO,CAACkE,GAAG,CAACC,gBAAK,CAACC,IAAI,CAAC,aAAa,EAAEH,OAAO,CAACd,MAAM,CAACV,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;;EAEhE;EACA;EACA;EACA,IAAIwB,OAAO,CAACf,MAAM,KAAK,QAAQ,IAAIe,OAAO,CAACvD,mBAAmB,EAAE2B,MAAM,EAAE;IACtErC,sBAAO,CAACkE,GAAG,CAACC,gBAAK,CAACE,MAAM,CAAC,yBAAyB,EAAEJ,OAAO,CAACvD,mBAAmB,CAAC+B,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;EAC7F;EAEA,OAAOwB,OAAO;AAChB;;AAEA;;AAEA,IAAIK,IAA6C,GAAG,IAAI;;AAExD;AACA;AACA;AACA;AACA;AACA;AACO,SAAStF,GAAGA,CACjByE,WAAmB,EACnB;EACER,KAAK;EACLnD;AAIF,CAAC,GAAG,CAAC,CAAC,EACN;EACA,IAAI,CAACV,SAAS,CAAC,CAAC,EAAE;IAChBjB,KAAK,CAAC,uDAAuD,CAAC;IAC9D,OAAO;MAAEyB,GAAG,EAAE,CAAC,CAAC;MAAEa,KAAK,EAAE,EAAE;MAAEC,mBAAmB,EAAE;IAAe,CAAC;EACpE;EACA,IAAIuC,KAAK,IAAI,CAACqB,IAAI,EAAE;IAClBA,IAAI,GAAGd,eAAe,CAACC,WAAW,EAAE;MAAE3D;IAAO,CAAC,CAAC;EACjD;EACA,OAAOwE,IAAI;AACb;;AAEA;AACA;AACA;AACA;AACA;AACO,SAASC,IAAIA,CAClBd,WAAmB,EACnBC,OAGC,GAAG,CAAC,CAAC,EACN;EACA,IAAI,CAACtE,SAAS,CAAC,CAAC,EAAE;IAChBjB,KAAK,CAAC,uDAAuD,CAAC;IAC9D,OAAOwB,OAAO,CAACC,GAAG;EACpB;EAEA,MAAMqE,OAAO,GAAGjF,GAAG,CAACyE,WAAW,EAAEC,OAAO,CAAC;EACzC,MAAMJ,aAAuB,GAAG,EAAE;EAElC,KAAK,MAAMzE,GAAG,IAAIoF,OAAO,CAACrE,GAAG,EAAE;IAC7B,IAAI,OAAOD,OAAO,CAACC,GAAG,CAACf,GAAG,CAAC,KAAK,WAAW,EAAE;MAC3CV,KAAK,CAAC,IAAIU,GAAG,6CAA6C,CAAC;IAC7D,CAAC,MAAM;MACL;MACAF,gBAAgB,CAACgB,OAAO,CAACC,GAAG,EAAEf,GAAG,CAAC;MAClCc,OAAO,CAACC,GAAG,CAACf,GAAG,CAAC,GAAGoF,OAAO,CAACrE,GAAG,CAACf,GAAG,CAAC;MACnCyE,aAAa,CAACzB,IAAI,CAAChD,GAAG,CAAC;IACzB;EACF;;EAEA;EACAmF,YAAY,CAAC;IAAE,GAAGC,OAAO;IAAEf,MAAM,EAAE,QAAQ;IAAEC,MAAM,EAAEG;EAAc,CAAC,EAAEI,OAAO,CAAC;EAE9E,OAAO/D,OAAO,CAACC,GAAG;AACpB;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS4E,QAAQA,CAAC9E,IAAwB,EAAE;EAAEI,MAAM,GAAG;AAA4B,CAAC,GAAG,CAAC,CAAC,EAAE;EAChG,IAAI,CAACV,SAAS,CAAC,CAAC,EAAE;IAChBjB,KAAK,CAAC,uDAAuD,CAAC;IAC9D,OAAO,EAAE;EACX;EAEA,OAAOsB,WAAW,CAAC;IAAEC,IAAI;IAAEI;EAAO,CAAC,CAAC;AACtC;;AAEA;AACA;AACA;AACO,SAAS2E,QAAQA,CAACC,QAAgB,EAAEC,SAAqB,EAAa;EAC3E,IAAI;IACF,MAAM/E,GAAG,GAAG,IAAA4B,cAAK,EAACkD,QAAQ,CAAC;IAC3B,KAAK,MAAM7F,GAAG,IAAIe,GAAG,EAAE;MACrB,IAAI,IAAAoC,4BAAe,EAACnD,GAAG,CAAC,EAAE;QACxB,OAAOe,GAAG,CAACf,GAAG,CAAC;MACjB;IACF;IACA,OAAO,IAAA6D,eAAM,EAAC9C,GAAG,EAAE+E,SAAS,IAAI,CAAC,CAAC,CAAC;EACrC,CAAC,CAAC,MAAM;IACN,OAAO,CAAC,CAAC;EACX;AACF","ignoreList":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@expo/env",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.3.0",
|
|
4
4
|
"description": "hydrate environment variables from .env files into process.env",
|
|
5
5
|
"main": "build/index.js",
|
|
6
6
|
"repository": {
|
|
@@ -29,12 +29,12 @@
|
|
|
29
29
|
"@babel/cli": "^7.23.4",
|
|
30
30
|
"@types/debug": "^4.1.7",
|
|
31
31
|
"@types/getenv": "^1.0.0",
|
|
32
|
-
"expo-module-scripts": "56.0.
|
|
32
|
+
"expo-module-scripts": "56.0.2"
|
|
33
33
|
},
|
|
34
34
|
"publishConfig": {
|
|
35
35
|
"access": "public"
|
|
36
36
|
},
|
|
37
|
-
"gitHead": "
|
|
37
|
+
"gitHead": "c4c9867a0bcbb188e55ecaec4998e38d33108a5d",
|
|
38
38
|
"scripts": {
|
|
39
39
|
"build": "tsc --emitDeclarationOnly && babel src --out-dir build --extensions \".ts\" --source-maps --ignore \"src/**/__mocks__/*\",\"src/**/__tests__/*\"",
|
|
40
40
|
"clean": "expo-module clean",
|