@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.
@@ -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: (0, _parse().expand)(loadedEnvVars, systemEnv),
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 {};
@@ -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":"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","ignoreList":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@expo/env",
3
- "version": "2.2.2-canary-20260506-03817f5",
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.1"
32
+ "expo-module-scripts": "56.0.2"
33
33
  },
34
34
  "publishConfig": {
35
35
  "access": "public"
36
36
  },
37
- "gitHead": "03817f532b50afebd227eb8eb15ede60fdbb4c6b",
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",