Fix env capture (#5205)

* Fix env capture

* Add test for env capture
This commit is contained in:
Tomoki Aonuma 2022-04-16 07:38:27 +09:00 committed by GitHub
parent 5bf1c98a39
commit c17129a92a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 83 additions and 19 deletions

View File

@ -1,7 +1,7 @@
use crate::CliError; use crate::CliError;
use log::trace; use log::trace;
use nu_engine::eval_block; use nu_engine::eval_block;
use nu_parser::{lex, parse, trim_quotes, Token, TokenContents}; use nu_parser::{lex, parse, unescape_unquote_string, Token, TokenContents};
use nu_protocol::engine::StateWorkingSet; use nu_protocol::engine::StateWorkingSet;
use nu_protocol::{ use nu_protocol::{
ast::Call, ast::Call,
@ -98,6 +98,10 @@ pub fn print_pipeline_data(
// env vars into it (in a "NAME"="value" format, quite similar to the output of the Unix 'env' // env vars into it (in a "NAME"="value" format, quite similar to the output of the Unix 'env'
// tool), then uses the file to get the spans. The file stays in memory, no filesystem IO is done. // tool), then uses the file to get the spans. The file stays in memory, no filesystem IO is done.
pub fn gather_parent_env_vars(engine_state: &mut EngineState) { pub fn gather_parent_env_vars(engine_state: &mut EngineState) {
gather_env_vars(std::env::vars(), engine_state);
}
fn gather_env_vars(vars: impl Iterator<Item = (String, String)>, engine_state: &mut EngineState) {
fn report_capture_error(engine_state: &EngineState, env_str: &str, msg: &str) { fn report_capture_error(engine_state: &EngineState, env_str: &str, msg: &str) {
let working_set = StateWorkingSet::new(engine_state); let working_set = StateWorkingSet::new(engine_state);
report_error( report_error(
@ -110,20 +114,35 @@ pub fn gather_parent_env_vars(engine_state: &mut EngineState) {
} }
fn put_env_to_fake_file(name: &str, val: &str, fake_env_file: &mut String) { fn put_env_to_fake_file(name: &str, val: &str, fake_env_file: &mut String) {
fake_env_file.push('`'); fn push_string_literal(s: &str, fake_env_file: &mut String) {
fake_env_file.push_str(name); fake_env_file.push('"');
fake_env_file.push('`'); for c in s.chars() {
if c == '\\' || c == '"' {
fake_env_file.push('\\');
}
fake_env_file.push(c);
}
fake_env_file.push('"');
}
push_string_literal(name, fake_env_file);
fake_env_file.push('='); fake_env_file.push('=');
fake_env_file.push('`'); push_string_literal(val, fake_env_file);
fake_env_file.push_str(val);
fake_env_file.push('`');
fake_env_file.push('\n'); fake_env_file.push('\n');
} }
let mut fake_env_file = String::new(); let mut fake_env_file = String::new();
let mut has_pwd = false;
// Make sure we always have PWD // Write all the env vars into a fake file
if std::env::var("PWD").is_err() { for (name, val) in vars {
if name == "PWD" {
has_pwd = true;
}
put_env_to_fake_file(&name, &val, &mut fake_env_file);
}
if !has_pwd {
match std::env::current_dir() { match std::env::current_dir() {
Ok(cwd) => { Ok(cwd) => {
put_env_to_fake_file("PWD", &cwd.to_string_lossy(), &mut fake_env_file); put_env_to_fake_file("PWD", &cwd.to_string_lossy(), &mut fake_env_file);
@ -142,11 +161,6 @@ pub fn gather_parent_env_vars(engine_state: &mut EngineState) {
} }
} }
// Write all the env vars into a fake file
for (name, val) in std::env::vars() {
put_env_to_fake_file(&name, &val, &mut fake_env_file);
}
// Lex the fake file, assign spans to all environment variables and add them // Lex the fake file, assign spans to all environment variables and add them
// to stack // to stack
let span_offset = engine_state.next_span_start(); let span_offset = engine_state.next_span_start();
@ -184,8 +198,19 @@ pub fn gather_parent_env_vars(engine_state: &mut EngineState) {
continue; continue;
} }
let bytes = trim_quotes(bytes); let (bytes, parse_error) = unescape_unquote_string(bytes, *span);
String::from_utf8_lossy(bytes).to_string()
if parse_error.is_some() {
report_capture_error(
engine_state,
&String::from_utf8_lossy(contents),
"Got unparsable name.",
);
continue;
}
bytes
} else { } else {
report_capture_error( report_capture_error(
engine_state, engine_state,
@ -213,10 +238,20 @@ pub fn gather_parent_env_vars(engine_state: &mut EngineState) {
continue; continue;
} }
let bytes = trim_quotes(bytes); let (bytes, parse_error) = unescape_unquote_string(bytes, *span);
if parse_error.is_some() {
report_capture_error(
engine_state,
&String::from_utf8_lossy(contents),
"Got unparsable value.",
);
continue;
}
Value::String { Value::String {
val: String::from_utf8_lossy(bytes).to_string(), val: bytes,
span: *span, span: *span,
} }
} else { } else {
@ -350,3 +385,32 @@ pub fn get_init_cwd() -> PathBuf {
}, },
} }
} }
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_gather_env_vars() {
let mut engine_state = EngineState::new();
let symbols = r##" !"#$%&'()*+,-./:;<=>?@[\]^_`{|}~"##;
gather_env_vars(
[
("FOO".into(), "foo".into()),
("SYMBOLS".into(), symbols.into()),
(symbols.into(), "symbols".into()),
]
.into_iter(),
&mut engine_state,
);
let env = engine_state.env_vars;
assert!(matches!(env.get("FOO"), Some(Value::String { val, .. }) if val == "foo"));
assert!(matches!(env.get("SYMBOLS"), Some(Value::String { val, .. }) if val == symbols));
assert!(matches!(env.get(symbols), Some(Value::String { val, .. }) if val == "symbols"));
assert!(env.get("PWD").is_some());
assert_eq!(env.len(), 4);
}
}

View File

@ -15,7 +15,7 @@ pub use lite_parse::{lite_parse, LiteBlock};
pub use parser::{ pub use parser::{
is_math_expression_like, parse, parse_block, parse_duration_bytes, parse_external_call, is_math_expression_like, parse, parse_block, parse_duration_bytes, parse_external_call,
trim_quotes, Import, trim_quotes, unescape_unquote_string, Import,
}; };
#[cfg(feature = "plugin")] #[cfg(feature = "plugin")]