nushell/crates/nu-protocol/src/engine/stack.rs

143 lines
4.5 KiB
Rust
Raw Normal View History

2021-10-25 16:58:58 +00:00
use std::collections::HashMap;
2021-08-15 22:33:34 +00:00
use crate::{Config, ShellError, Value, VarId, CONFIG_VARIABLE_ID};
2021-08-15 22:33:34 +00:00
/// A runtime value stack used during evaluation
///
/// A note on implementation:
///
/// We previously set up the stack in a traditional way, where stack frames had parents which would
/// represent other frames that you might return to when exiting a function.
///
/// While experimenting with blocks, we found that we needed to have closure captures of variables
/// seen outside of the blocks, so that they blocks could be run in a way that was both thread-safe
/// and followed the restrictions for closures applied to iterators. The end result left us with
/// closure-captured single stack frames that blocks could see.
///
/// Blocks make up the only scope and stack definition abstraction in Nushell. As a result, we were
/// creating closure captures at any point we wanted to have a Block value we could safely evaluate
/// in any context. This meant that the parents were going largely unused, with captured variables
/// taking their place. The end result is this, where we no longer have separate frames, but instead
/// use the Stack as a way of representing the local and closure-captured state.
2021-10-25 04:01:02 +00:00
#[derive(Debug, Clone)]
2021-10-25 20:04:23 +00:00
pub struct Stack {
/// Variables
2021-08-15 22:33:34 +00:00
pub vars: HashMap<VarId, Value>,
/// Environment variables arranged as a stack to be able to recover values from parent scopes
pub env_vars: Vec<HashMap<String, String>>,
2021-08-15 22:33:34 +00:00
}
impl Default for Stack {
fn default() -> Self {
Self::new()
}
}
impl Stack {
pub fn new() -> Stack {
2021-10-25 20:04:23 +00:00
Stack {
2021-08-15 22:33:34 +00:00
vars: HashMap::new(),
env_vars: vec![],
2021-10-25 20:04:23 +00:00
}
2021-08-15 22:33:34 +00:00
}
2021-08-15 22:33:34 +00:00
pub fn get_var(&self, var_id: VarId) -> Result<Value, ShellError> {
2021-10-25 20:04:23 +00:00
if let Some(v) = self.vars.get(&var_id) {
return Ok(v.clone());
2021-08-15 22:33:34 +00:00
}
2021-12-02 23:11:25 +00:00
Err(ShellError::NushellFailed("variable not found".into()))
2021-08-15 22:33:34 +00:00
}
2021-10-25 04:01:02 +00:00
pub fn add_var(&mut self, var_id: VarId, value: Value) {
2021-10-25 20:04:23 +00:00
self.vars.insert(var_id, value);
2021-08-15 22:33:34 +00:00
}
2021-10-25 04:01:02 +00:00
pub fn add_env_var(&mut self, var: String, value: String) {
if let Some(scope) = self.env_vars.last_mut() {
scope.insert(var, value);
} else {
self.env_vars.push(HashMap::from([(var, value)]));
}
2021-08-15 22:33:34 +00:00
}
2021-10-25 20:04:23 +00:00
pub fn collect_captures(&self, captures: &[VarId]) -> Stack {
let mut output = Stack::new();
for capture in captures {
2021-10-25 21:14:21 +00:00
// Note: this assumes we have calculated captures correctly and that commands
// that take in a var decl will manually set this into scope when running the blocks
if let Ok(value) = self.get_var(*capture) {
output.vars.insert(*capture, value);
}
2021-10-25 20:04:23 +00:00
}
2021-10-25 04:01:02 +00:00
2021-11-04 02:32:35 +00:00
// FIXME: this is probably slow
output.env_vars = self.env_vars.clone();
output.env_vars.push(HashMap::new());
2021-11-04 02:32:35 +00:00
let config = self
.get_var(CONFIG_VARIABLE_ID)
.expect("internal error: config is missing");
output.vars.insert(CONFIG_VARIABLE_ID, config);
2021-10-25 04:01:02 +00:00
output
2021-08-15 22:33:34 +00:00
}
/// Flatten the env var scope frames into one frame
2021-09-19 19:29:58 +00:00
pub fn get_env_vars(&self) -> HashMap<String, String> {
let mut result = HashMap::new();
for scope in &self.env_vars {
result.extend(scope.clone());
}
result
2021-09-19 19:29:58 +00:00
}
2021-10-02 13:10:28 +00:00
pub fn get_env_var(&self, name: &str) -> Option<String> {
for scope in self.env_vars.iter().rev() {
if let Some(v) = scope.get(name) {
return Some(v.to_string());
}
2021-10-25 04:01:02 +00:00
}
2021-10-25 04:01:02 +00:00
None
2021-10-02 13:10:28 +00:00
}
pub fn remove_env_var(&mut self, name: &str) -> Option<String> {
for scope in self.env_vars.iter_mut().rev() {
if let Some(v) = scope.remove(name) {
return Some(v);
}
}
None
}
pub fn get_config(&self) -> Result<Config, ShellError> {
let config = self.get_var(CONFIG_VARIABLE_ID);
match config {
Ok(config) => config.into_config(),
Err(e) => {
println!("Can't find {} in {:?}", CONFIG_VARIABLE_ID, self);
Err(e)
}
}
}
2021-08-15 22:33:34 +00:00
pub fn print_stack(&self) {
2021-10-25 20:04:23 +00:00
println!("vars:");
for (var, val) in &self.vars {
println!(" {}: {:?}", var, val);
}
for (i, scope) in self.env_vars.iter().rev().enumerate() {
println!("env vars, scope {} (from the last);", i);
for (var, val) in scope {
println!(" {}: {:?}", var, val);
}
2021-08-15 22:33:34 +00:00
}
}
}