Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

BREAKING(dotenv): remove defaultsPath option from load[Sync]() #5451

Merged
merged 4 commits into from
Jul 19, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
66 changes: 4 additions & 62 deletions dotenv/mod.ts
Original file line number Diff line number Diff line change
Expand Up @@ -58,21 +58,6 @@ export interface LoadOptions {
* @default {false}
*/
allowEmptyValues?: boolean;

/**
* Optional path to `.env.defaults` file which is used to define default
* (fallback) values. To prevent the default value from being used,
* set to `null`.
*
* ```sh
* # .env.defaults
* # Will not be set if GREETING is set in base .env file
* GREETING="a secret to everybody"
* ```
*
* @default {"./.env.defaults"}
*/
defaultsPath?: string | null;
}

/**
Expand All @@ -94,21 +79,11 @@ export function loadSync(
const {
envPath = ".env",
examplePath = ".env.example",
defaultsPath = ".env.defaults",
export: _export = false,
allowEmptyValues = false,
} = options;
const conf = envPath ? parseFileSync(envPath) : {};

if (defaultsPath) {
const confDefaults = parseFileSync(defaultsPath);
for (const [key, value] of Object.entries(confDefaults)) {
if (!(key in conf)) {
conf[key] = value;
}
}
}

if (examplePath) {
const confExample = parseFileSync(examplePath);
assertSafe(conf, confExample, allowEmptyValues);
Expand Down Expand Up @@ -176,15 +151,14 @@ export function loadSync(
* |----|-------|
* |.env|primary file for storing key-value environment entries
* |.env.example|this file does not set any values, but specifies env variables which must be present in the configuration object or process environment after loading dotenv
* |.env.defaults|specify default values for env variables to be used when there is no entry in the `.env` file
*
* ### Example file
*
* The purpose of the example file is to provide a list of environment
* variables which must be set or already present in the process environment
* or an exception will be thrown. These
* variables may be set externally or loaded via the `.env` or
* `.env.defaults` files. A description may also be provided to help
* variables may be set externally or loaded via the `.env` files.
* A description may also be provided to help
* understand the purpose of the env variable. The values in this file
* are for documentation only and are not set in the environment. Example:
*
Expand All @@ -202,27 +176,6 @@ export function loadSync(
* DATA_KEY or DATA_URL is not present in the environment an exception
* is thrown.
*
* ### Defaults
*
* This file is used to provide a list of default environment variables
* which will be used if there is no overriding variable in the `.env`
* file.
*
* ```sh
* # .env.defaults
* KEY_1=DEFAULT_VALUE
* KEY_2=ANOTHER_DEFAULT_VALUE
* ```
* ```sh
* # .env
* KEY_1=ABCD
* ```
* The environment variables set after dotenv loads are:
* ```sh
* KEY_1=ABCD
* KEY_2=ANOTHER_DEFAULT_VALUE
* ```
*
* ## Configuration
*
* Loading environment files comes with a number of options passed into
Expand All @@ -231,9 +184,8 @@ export function loadSync(
* |Option|Default|Description
* |------|-------|-----------
* |envPath|./.env|Path and filename of the `.env` file. Use null to prevent the .env file from being loaded.
* |defaultsPath|./.env.defaults|Path and filename of the `.env.defaults` file. Use null to prevent the .env.defaults file from being loaded.
* |examplePath|./.env.example|Path and filename of the `.env.example` file. Use null to prevent the .env.example file from being loaded.
* |export|false|When true, this will export all environment variables in the `.env` and `.env.default` files to the process environment (e.g. for use by `Deno.env.get()`) but only if they are not already set. If a variable is already in the process, the `.env` value is ignored.
* |export|false|When true, this will export all environment variables in the `.env` file to the process environment (e.g. for use by `Deno.env.get()`) but only if they are not already set. If a variable is already in the process, the `.env` value is ignored.
* |allowEmptyValues|false|Allows empty values for specified env variables (throws otherwise)
*
* ### Example configuration
Expand All @@ -257,7 +209,7 @@ export function loadSync(
* in your `.env` file, you will need the `--allow-env` permission. E.g.
*
* ```sh
* deno run --allow-read=.env,.env.defaults,.env.example --allow-env=ENV1,ENV2 app.ts
* deno run --allow-read=.env,.env.example --allow-env=ENV1,ENV2 app.ts
* ```
*
* ## Parsing Rules
Expand Down Expand Up @@ -309,21 +261,11 @@ export async function load(
const {
envPath = ".env",
examplePath = ".env.example",
defaultsPath = ".env.defaults",
export: _export = false,
allowEmptyValues = false,
} = options;
const conf = envPath ? await parseFile(envPath) : {};

if (defaultsPath) {
const confDefaults = await parseFile(defaultsPath);
for (const [key, value] of Object.entries(confDefaults)) {
if (!(key in conf)) {
conf[key] = value;
}
}
}

if (examplePath) {
const confExample = await parseFile(examplePath);
assertSafe(conf, confExample, allowEmptyValues);
Expand Down
92 changes: 3 additions & 89 deletions dotenv/mod_test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,43 +20,27 @@ const testdataDir = path.resolve(moduleDir, "testdata");

const testOptions = Object.freeze({
envPath: path.join(testdataDir, ".env"),
defaultsPath: path.join(testdataDir, ".env.defaults"),
});

Deno.test("load() handles non-existent .env files", async () => {
//n.b. neither .env nor .env.default exist in the current directory
// .env doesn't exist in the current directory
assertEquals({}, await load());
assertEquals({}, loadSync());

const loadOptions = {
envPath: "some.nonexistent.env",
examplePath: "some.nonexistent.example",
defaultsPath: "some.nonexistent.defaults",
};
assertEquals({}, await load(loadOptions));
assertEquals({}, loadSync(loadOptions));
});

Deno.test("load() handles build from .env.default only", async () => {
const conf = loadSync({
defaultsPath: path.join(testdataDir, ".env.defaults"),
});
assertEquals(conf.DEFAULT1, "Some Default", "loaded from .env.default");

const asyncConf = await load({
defaultsPath: path.join(testdataDir, ".env.defaults"),
});
assertEquals(asyncConf.DEFAULT1, "Some Default", "loaded from .env.default");
});

Deno.test("load() handles comprised .env and .env.defaults", async () => {
const conf = loadSync(testOptions);
assertEquals(conf.GREETING, "hello world", "loaded from .env");
assertEquals(conf.DEFAULT1, "Some Default", "loaded from .env.default");

const asyncConf = await load(testOptions);
assertEquals(asyncConf.GREETING, "hello world", "loaded from .env");
assertEquals(asyncConf.DEFAULT1, "Some Default", "loaded from .env.default");
});

Deno.test("load() handles exported entries accessibility in Deno.env", async () => {
Expand All @@ -77,14 +61,8 @@ function validateExport(): void {
"hello world",
"exported from .env -> Deno.env",
);
assertEquals(
Deno.env.get("DEFAULT1"),
"Some Default",
"exported from .env.default -> Deno.env",
);
} finally {
Deno.env.delete("GREETING");
Deno.env.delete("DEFAULT1");
}
}

Expand All @@ -104,11 +82,6 @@ function validateNotOverridden(conf: Record<string, string>): void {
"Do not override!",
"not exported from .env -> Deno.env",
);
assertEquals(
Deno.env.get("DEFAULT1"),
"Some Default",
"exported from .env.default -> Deno.env",
);
} finally {
Deno.env.delete("DEFAULT1");
}
Expand All @@ -124,16 +97,6 @@ Deno.test("load() handles example file key in .env, no issues loading", async ()
await load(loadOptions);
});

Deno.test("load() handles example file key in .env.default, no issues loading", async () => {
//Both .env.example3.test and .env.default contain "DEFAULT1"
const loadOptions = {
...testOptions,
examplePath: path.join(testdataDir, "./.env.example3.test"),
};
loadSync(loadOptions);
await load(loadOptions);
});

Deno.test("load() handles example file key not in .env or .env.defaults, error thrown", async () => {
// Example file key of "ANOTHER" is not present in .env or .env.defaults
const error: MissingEnvVarsError = assertThrows(() => {
Expand Down Expand Up @@ -249,26 +212,6 @@ Deno.test("load() checks that required keys sourced from process environment can
}
});

Deno.test("load() loads .env and .env.defaults successfully from default file names/paths", async () => {
const command = new Deno.Command(Deno.execPath(), {
args: [
"run",
"--no-lock",
"--allow-read",
"--allow-env",
path.join(testdataDir, "./app_defaults.ts"),
],
cwd: testdataDir,
});
const { stdout } = await command.output();

const decoder = new TextDecoder();
const conf = JSON.parse(decoder.decode(stdout).trim());

assertEquals(conf.GREETING, "hello world", "fetches .env by default");
assertEquals(conf.DEFAULT1, "Some Default", "default value loaded");
});

Deno.test("load() expands empty values from process env expand as empty value", async () => {
try {
Deno.env.set("EMPTY", "");
Expand Down Expand Up @@ -308,7 +251,6 @@ Deno.test(
// note lack of --allow-env permission
const conf = loadSync(testOptions);
assertEquals(conf.GREETING, "hello world");
assertEquals(conf.DEFAULT1, "Some Default");
},
);

Expand All @@ -324,7 +266,6 @@ Deno.test(
// note lack of --allow-env permission
const loadOptions = {
envPath: path.join(testdataDir, "./.env.single.expand"),
defaultsPath: null,
examplePath: null,
};
assertThrows(
Expand All @@ -349,7 +290,6 @@ Deno.test(

const loadOptions = {
envPath: path.join(testdataDir, "./.env.single.expand"),
defaultsPath: null,
examplePath: null,
};
const conf = loadSync(loadOptions);
Expand All @@ -376,7 +316,6 @@ Deno.test(
},
async (t) => {
const optsNoPaths = {
defaultsPath: null,
envPath: null,
examplePath: null,
} satisfies LoadOptions;
Expand All @@ -387,7 +326,6 @@ Deno.test(

const optsOnlyEnvPath = {
...optsEnvPath,
defaultsPath: null,
examplePath: null,
} satisfies LoadOptions;

Expand All @@ -402,45 +340,21 @@ Deno.test(
assertEnv(await load(optsOnlyEnvPath));

await assertRejects(
() => load(optsEnvPath),
Deno.errors.PermissionDenied,
`Requires read access to ".env.defaults"`,
);

await assertRejects(
() => load({ ...optsEnvPath, defaultsPath: null }),
() => load({ ...optsEnvPath }),
Deno.errors.PermissionDenied,
`Requires read access to ".env.example"`,
);

await assertRejects(
() => load({ ...optsEnvPath, examplePath: null }),
Deno.errors.PermissionDenied,
`Requires read access to ".env.defaults"`,
);
});

await t.step("loadSync", () => {
assertStrictEquals(Object.keys(loadSync(optsNoPaths)).length, 0);
assertEnv(loadSync(optsOnlyEnvPath));

assertThrows(
() => loadSync(optsEnvPath),
Deno.errors.PermissionDenied,
`Requires read access to ".env.defaults"`,
);

assertThrows(
() => loadSync({ ...optsEnvPath, defaultsPath: null }),
() => loadSync({ ...optsEnvPath }),
Deno.errors.PermissionDenied,
`Requires read access to ".env.example"`,
);

assertThrows(
() => loadSync({ ...optsEnvPath, examplePath: null }),
Deno.errors.PermissionDenied,
`Requires read access to ".env.defaults"`,
);
});
},
);