mirror of
https://github.com/nodejs/node.git
synced 2025-04-29 14:25:18 +00:00
util: fix parseEnv incorrectly splitting multiple ‘=‘ in value
Previously, parseEnv would create multiple environment variables if a single line contained multiple ‘=‘ characters (e.g. A=B=C would become { A: ‘B=C’, B: ‘C’ }). This commit ensures that only the first ‘=‘ is used as the key-value delimiter, and the rest of the line is treated as the value. Fixes: https://github.com/nodejs/node/issues/57411 PR-URL: https://github.com/nodejs/node/pull/57421 Reviewed-By: Daniel Lemire <daniel@lemire.me> Reviewed-By: Matteo Collina <matteo.collina@gmail.com>
This commit is contained in:
parent
74722a55a6
commit
e6a0d77d57
@ -6,6 +6,8 @@ BASIC=basic
|
|||||||
|
|
||||||
# previous line intentionally left blank
|
# previous line intentionally left blank
|
||||||
AFTER_LINE=after_line
|
AFTER_LINE=after_line
|
||||||
|
A="B=C"
|
||||||
|
B=C=D
|
||||||
EMPTY=
|
EMPTY=
|
||||||
EMPTY_SINGLE_QUOTES=''
|
EMPTY_SINGLE_QUOTES=''
|
||||||
EMPTY_DOUBLE_QUOTES=""
|
EMPTY_DOUBLE_QUOTES=""
|
||||||
|
@ -145,7 +145,16 @@ void Dotenv::ParseContent(const std::string_view input) {
|
|||||||
// If there is no equal character, then ignore everything
|
// If there is no equal character, then ignore everything
|
||||||
auto equal = content.find('=');
|
auto equal = content.find('=');
|
||||||
if (equal == std::string_view::npos) {
|
if (equal == std::string_view::npos) {
|
||||||
break;
|
auto newline = content.find('\n');
|
||||||
|
if (newline != std::string_view::npos) {
|
||||||
|
// If we used `newline` only,
|
||||||
|
// the '\n' might remain and cause an empty-line parse
|
||||||
|
content.remove_prefix(newline + 1);
|
||||||
|
} else {
|
||||||
|
content = {};
|
||||||
|
}
|
||||||
|
// No valid data here, skip to next line
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
key = content.substr(0, equal);
|
key = content.substr(0, equal);
|
||||||
@ -195,7 +204,9 @@ void Dotenv::ParseContent(const std::string_view input) {
|
|||||||
store_.insert_or_assign(std::string(key), multi_line_value);
|
store_.insert_or_assign(std::string(key), multi_line_value);
|
||||||
auto newline = content.find('\n', closing_quote + 1);
|
auto newline = content.find('\n', closing_quote + 1);
|
||||||
if (newline != std::string_view::npos) {
|
if (newline != std::string_view::npos) {
|
||||||
content.remove_prefix(newline);
|
content.remove_prefix(newline + 1);
|
||||||
|
} else {
|
||||||
|
content = {};
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -216,7 +227,7 @@ void Dotenv::ParseContent(const std::string_view input) {
|
|||||||
if (newline != std::string_view::npos) {
|
if (newline != std::string_view::npos) {
|
||||||
value = content.substr(0, newline);
|
value = content.substr(0, newline);
|
||||||
store_.insert_or_assign(std::string(key), value);
|
store_.insert_or_assign(std::string(key), value);
|
||||||
content.remove_prefix(newline);
|
content.remove_prefix(newline + 1);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Example: KEY="value"
|
// Example: KEY="value"
|
||||||
@ -226,8 +237,13 @@ void Dotenv::ParseContent(const std::string_view input) {
|
|||||||
// since there could be newline characters inside the value.
|
// since there could be newline characters inside the value.
|
||||||
auto newline = content.find('\n', closing_quote + 1);
|
auto newline = content.find('\n', closing_quote + 1);
|
||||||
if (newline != std::string_view::npos) {
|
if (newline != std::string_view::npos) {
|
||||||
content.remove_prefix(newline);
|
// Use +1 to discard the '\n' itself => next line
|
||||||
|
content.remove_prefix(newline + 1);
|
||||||
|
} else {
|
||||||
|
content = {};
|
||||||
}
|
}
|
||||||
|
// No valid data here, skip to next line
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Regular key value pair.
|
// Regular key value pair.
|
||||||
@ -243,15 +259,21 @@ void Dotenv::ParseContent(const std::string_view input) {
|
|||||||
if (hash_character != std::string_view::npos) {
|
if (hash_character != std::string_view::npos) {
|
||||||
value = content.substr(0, hash_character);
|
value = content.substr(0, hash_character);
|
||||||
}
|
}
|
||||||
content.remove_prefix(newline);
|
store_.insert_or_assign(std::string(key), trim_spaces(value));
|
||||||
|
content.remove_prefix(newline + 1);
|
||||||
} else {
|
} else {
|
||||||
// In case the last line is a single key/value pair
|
// In case the last line is a single key/value pair
|
||||||
// Example: KEY=VALUE (without a newline at the EOF)
|
// Example: KEY=VALUE (without a newline at the EOF)
|
||||||
value = content.substr(0);
|
value = content;
|
||||||
|
auto hash_char = value.find('#');
|
||||||
|
if (hash_char != std::string_view::npos) {
|
||||||
|
value = content.substr(0, hash_char);
|
||||||
|
}
|
||||||
|
store_.insert_or_assign(std::string(key), trim_spaces(value));
|
||||||
|
content = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
value = trim_spaces(value);
|
store_.insert_or_assign(std::string(key), trim_spaces(value));
|
||||||
store_.insert_or_assign(std::string(key), value);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
2
test/fixtures/dotenv/valid.env
vendored
2
test/fixtures/dotenv/valid.env
vendored
@ -6,6 +6,8 @@ BASIC=basic
|
|||||||
|
|
||||||
# previous line intentionally left blank
|
# previous line intentionally left blank
|
||||||
AFTER_LINE=after_line
|
AFTER_LINE=after_line
|
||||||
|
A="B=C"
|
||||||
|
B=C=D
|
||||||
EMPTY=
|
EMPTY=
|
||||||
EMPTY_SINGLE_QUOTES=''
|
EMPTY_SINGLE_QUOTES=''
|
||||||
EMPTY_DOUBLE_QUOTES=""
|
EMPTY_DOUBLE_QUOTES=""
|
||||||
|
@ -82,3 +82,5 @@ assert.strictEqual(process.env.DONT_EXPAND_SQUOTED, 'dontexpand\\nnewlines');
|
|||||||
assert.strictEqual(process.env.EXPORT_EXAMPLE, 'ignore export');
|
assert.strictEqual(process.env.EXPORT_EXAMPLE, 'ignore export');
|
||||||
// Ignore spaces before double quotes to avoid quoted strings as value
|
// Ignore spaces before double quotes to avoid quoted strings as value
|
||||||
assert.strictEqual(process.env.SPACE_BEFORE_DOUBLE_QUOTES, 'space before double quotes');
|
assert.strictEqual(process.env.SPACE_BEFORE_DOUBLE_QUOTES, 'space before double quotes');
|
||||||
|
assert.strictEqual(process.env.A, 'B=C');
|
||||||
|
assert.strictEqual(process.env.B, 'C=D');
|
||||||
|
@ -11,6 +11,8 @@ const fs = require('node:fs');
|
|||||||
const validContent = fs.readFileSync(validEnvFilePath, 'utf8');
|
const validContent = fs.readFileSync(validEnvFilePath, 'utf8');
|
||||||
|
|
||||||
assert.deepStrictEqual(util.parseEnv(validContent), {
|
assert.deepStrictEqual(util.parseEnv(validContent), {
|
||||||
|
A: 'B=C',
|
||||||
|
B: 'C=D',
|
||||||
AFTER_LINE: 'after_line',
|
AFTER_LINE: 'after_line',
|
||||||
BACKTICKS: 'backticks',
|
BACKTICKS: 'backticks',
|
||||||
BACKTICKS_INSIDE_DOUBLE: '`backticks` work inside double quotes',
|
BACKTICKS_INSIDE_DOUBLE: '`backticks` work inside double quotes',
|
||||||
|
Loading…
Reference in New Issue
Block a user