mirror of
https://github.com/nushell/nushell.git
synced 2024-12-27 01:19:25 +01:00
168 lines
5.3 KiB
Rust
168 lines
5.3 KiB
Rust
use log::trace;
|
|
use miette::{IntoDiagnostic, Result};
|
|
use nu_engine::{convert_env_values, eval_block};
|
|
use nu_parser::parse;
|
|
use nu_protocol::{
|
|
engine::{EngineState, StateDelta, StateWorkingSet},
|
|
Config, PipelineData, Span, Value, CONFIG_VARIABLE_ID,
|
|
};
|
|
use std::path::PathBuf;
|
|
|
|
use crate::utils::{gather_parent_env_vars, report_error};
|
|
|
|
/// Main function used when a file path is found as argument for nu
|
|
pub(crate) fn evaluate(
|
|
path: String,
|
|
init_cwd: PathBuf,
|
|
engine_state: &mut EngineState,
|
|
) -> Result<()> {
|
|
// First, set up env vars as strings only
|
|
gather_parent_env_vars(engine_state);
|
|
|
|
let file = std::fs::read(&path).into_diagnostic()?;
|
|
|
|
let (block, delta) = {
|
|
let mut working_set = StateWorkingSet::new(engine_state);
|
|
trace!("parsing file: {}", path);
|
|
|
|
let (output, err) = parse(&mut working_set, Some(&path), &file, false);
|
|
if let Some(err) = err {
|
|
report_error(&working_set, &err);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
(output, working_set.render())
|
|
};
|
|
|
|
if let Err(err) = engine_state.merge_delta(delta, None, &init_cwd) {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
report_error(&working_set, &err);
|
|
}
|
|
|
|
let mut stack = nu_protocol::engine::Stack::new();
|
|
|
|
// Set up our initial config to start from
|
|
stack.vars.insert(
|
|
CONFIG_VARIABLE_ID,
|
|
Value::Record {
|
|
cols: vec![],
|
|
vals: vec![],
|
|
span: Span { start: 0, end: 0 },
|
|
},
|
|
);
|
|
|
|
let config = match stack.get_config() {
|
|
Ok(config) => config,
|
|
Err(e) => {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
|
|
report_error(&working_set, &e);
|
|
Config::default()
|
|
}
|
|
};
|
|
|
|
// Merge the delta in case env vars changed in the config
|
|
match nu_engine::env::current_dir(engine_state, &stack) {
|
|
Ok(cwd) => {
|
|
if let Err(e) = engine_state.merge_delta(StateDelta::new(), Some(&mut stack), cwd) {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
report_error(&working_set, &e);
|
|
}
|
|
}
|
|
Err(e) => {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
report_error(&working_set, &e);
|
|
}
|
|
}
|
|
|
|
// Translate environment variables from Strings to Values
|
|
if let Some(e) = convert_env_values(engine_state, &stack, &config) {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
report_error(&working_set, &e);
|
|
std::process::exit(1);
|
|
}
|
|
|
|
match eval_block(
|
|
engine_state,
|
|
&mut stack,
|
|
&block,
|
|
PipelineData::new(Span::new(0, 0)), // Don't try this at home, 0 span is ignored
|
|
) {
|
|
Ok(pipeline_data) => {
|
|
for item in pipeline_data {
|
|
if let Value::Error { error } = item {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
|
|
report_error(&working_set, &error);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
println!("{}", item.into_string("\n", &config));
|
|
}
|
|
|
|
// Next, let's check if there are any flags we want to pass to the main function
|
|
let args: Vec<String> = std::env::args().skip(2).collect();
|
|
|
|
if args.is_empty() && engine_state.find_decl(b"main").is_none() {
|
|
return Ok(());
|
|
}
|
|
|
|
let args = format!("main {}", args.join(" ")).as_bytes().to_vec();
|
|
|
|
let (block, delta) = {
|
|
let mut working_set = StateWorkingSet::new(engine_state);
|
|
let (output, err) = parse(&mut working_set, Some("<cmdline>"), &args, false);
|
|
if let Some(err) = err {
|
|
report_error(&working_set, &err);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
(output, working_set.render())
|
|
};
|
|
|
|
let cwd = nu_engine::env::current_dir_str(engine_state, &stack)?;
|
|
|
|
if let Err(err) = engine_state.merge_delta(delta, Some(&mut stack), &cwd) {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
report_error(&working_set, &err);
|
|
}
|
|
|
|
match eval_block(
|
|
engine_state,
|
|
&mut stack,
|
|
&block,
|
|
PipelineData::new(Span::new(0, 0)), // Don't try this at home, 0 span is ignored
|
|
) {
|
|
Ok(pipeline_data) => {
|
|
for item in pipeline_data {
|
|
if let Value::Error { error } = item {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
|
|
report_error(&working_set, &error);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
println!("{}", item.into_string("\n", &config));
|
|
}
|
|
}
|
|
Err(err) => {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
|
|
report_error(&working_set, &err);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
}
|
|
}
|
|
Err(err) => {
|
|
let working_set = StateWorkingSet::new(engine_state);
|
|
|
|
report_error(&working_set, &err);
|
|
|
|
std::process::exit(1);
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|