2023-08-26 13:41:29 +00:00
|
|
|
use crate::{
|
2023-09-05 14:35:58 +00:00
|
|
|
ast::{
|
|
|
|
eval_operator, Bits, Block, Boolean, Call, Comparison, Expr, Expression, Math, Operator,
|
|
|
|
PipelineElement,
|
|
|
|
},
|
2023-09-01 06:18:55 +00:00
|
|
|
engine::{EngineState, StateWorkingSet},
|
2023-09-05 14:35:58 +00:00
|
|
|
record, HistoryFileFormat, PipelineData, Range, Record, ShellError, Span, Value,
|
2022-12-21 22:21:03 +00:00
|
|
|
};
|
2023-09-01 06:18:55 +00:00
|
|
|
use nu_system::os_info::{get_kernel_version, get_os_arch, get_os_family, get_os_name};
|
2023-09-12 06:06:56 +00:00
|
|
|
use std::path::{Path, PathBuf};
|
2023-09-01 06:18:55 +00:00
|
|
|
|
|
|
|
pub fn create_nu_constant(engine_state: &EngineState, span: Span) -> Result<Value, ShellError> {
|
2023-09-12 06:06:56 +00:00
|
|
|
fn canonicalize_path(engine_state: &EngineState, path: &Path) -> PathBuf {
|
2023-09-01 06:18:55 +00:00
|
|
|
let cwd = engine_state.current_work_dir();
|
|
|
|
|
|
|
|
if path.exists() {
|
|
|
|
match nu_path::canonicalize_with(path, cwd) {
|
|
|
|
Ok(canon_path) => canon_path,
|
2023-09-12 06:06:56 +00:00
|
|
|
Err(_) => path.to_owned(),
|
2023-09-01 06:18:55 +00:00
|
|
|
}
|
|
|
|
} else {
|
2023-09-12 06:06:56 +00:00
|
|
|
path.to_owned()
|
2023-09-01 06:18:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut record = Record::new();
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"default-config-dir",
|
|
|
|
if let Some(mut path) = nu_path::config_dir() {
|
|
|
|
path.push("nushell");
|
|
|
|
Value::string(path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not get config directory".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"config-path",
|
|
|
|
if let Some(path) = engine_state.get_config_path("config-path") {
|
|
|
|
let canon_config_path = canonicalize_path(engine_state, path);
|
|
|
|
Value::string(canon_config_path.to_string_lossy(), span)
|
|
|
|
} else if let Some(mut path) = nu_path::config_dir() {
|
|
|
|
path.push("nushell");
|
|
|
|
path.push("config.nu");
|
|
|
|
Value::string(path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not get config directory".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"env-path",
|
|
|
|
if let Some(path) = engine_state.get_config_path("env-path") {
|
|
|
|
let canon_env_path = canonicalize_path(engine_state, path);
|
|
|
|
Value::string(canon_env_path.to_string_lossy(), span)
|
|
|
|
} else if let Some(mut path) = nu_path::config_dir() {
|
|
|
|
path.push("nushell");
|
|
|
|
path.push("env.nu");
|
|
|
|
Value::string(path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not find environment path".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"history-path",
|
|
|
|
if let Some(mut path) = nu_path::config_dir() {
|
|
|
|
path.push("nushell");
|
|
|
|
match engine_state.config.history_file_format {
|
|
|
|
HistoryFileFormat::Sqlite => {
|
|
|
|
path.push("history.sqlite3");
|
|
|
|
}
|
|
|
|
HistoryFileFormat::PlainText => {
|
|
|
|
path.push("history.txt");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let canon_hist_path = canonicalize_path(engine_state, &path);
|
|
|
|
Value::string(canon_hist_path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not find history path".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"loginshell-path",
|
|
|
|
if let Some(mut path) = nu_path::config_dir() {
|
|
|
|
path.push("nushell");
|
|
|
|
path.push("login.nu");
|
|
|
|
let canon_login_path = canonicalize_path(engine_state, &path);
|
|
|
|
Value::string(canon_login_path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not find login shell path".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
#[cfg(feature = "plugin")]
|
|
|
|
{
|
|
|
|
record.push(
|
|
|
|
"plugin-path",
|
|
|
|
if let Some(path) = &engine_state.plugin_signatures {
|
|
|
|
let canon_plugin_path = canonicalize_path(engine_state, path);
|
|
|
|
Value::string(canon_plugin_path.to_string_lossy(), span)
|
2023-09-04 00:19:04 +00:00
|
|
|
} else if let Some(mut plugin_path) = nu_path::config_dir() {
|
|
|
|
// If there are no signatures, we should still populate the plugin path
|
|
|
|
plugin_path.push("nushell");
|
|
|
|
plugin_path.push("plugin.nu");
|
|
|
|
Value::string(plugin_path.to_string_lossy(), span)
|
2023-09-01 06:18:55 +00:00
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not get plugin signature location".into()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"home-path",
|
|
|
|
if let Some(path) = nu_path::home_dir() {
|
|
|
|
let canon_home_path = canonicalize_path(engine_state, &path);
|
|
|
|
Value::string(canon_home_path.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(ShellError::IOError("Could not get home path".into()), span)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push("temp-path", {
|
|
|
|
let canon_temp_path = canonicalize_path(engine_state, &std::env::temp_dir());
|
|
|
|
Value::string(canon_temp_path.to_string_lossy(), span)
|
|
|
|
});
|
|
|
|
|
|
|
|
record.push("pid", Value::int(std::process::id().into(), span));
|
|
|
|
|
|
|
|
record.push("os-info", {
|
|
|
|
let ver = get_kernel_version();
|
|
|
|
Value::record(
|
|
|
|
record! {
|
|
|
|
"name" => Value::string(get_os_name(), span),
|
|
|
|
"arch" => Value::string(get_os_arch(), span),
|
|
|
|
"family" => Value::string(get_os_family(), span),
|
|
|
|
"kernel_version" => Value::string(ver, span),
|
|
|
|
},
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
});
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"startup-time",
|
|
|
|
Value::duration(engine_state.get_startup_time(), span),
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"is-interactive",
|
|
|
|
Value::bool(engine_state.is_interactive, span),
|
|
|
|
);
|
|
|
|
|
|
|
|
record.push("is-login", Value::bool(engine_state.is_login, span));
|
|
|
|
|
|
|
|
record.push(
|
|
|
|
"current-exe",
|
|
|
|
if let Ok(current_exe) = std::env::current_exe() {
|
|
|
|
Value::string(current_exe.to_string_lossy(), span)
|
|
|
|
} else {
|
|
|
|
Value::error(
|
|
|
|
ShellError::IOError("Could not get current executable path".to_string()),
|
|
|
|
span,
|
|
|
|
)
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(Value::record(record, span))
|
|
|
|
}
|
2022-12-21 22:21:03 +00:00
|
|
|
|
2023-08-26 13:41:29 +00:00
|
|
|
fn eval_const_call(
|
|
|
|
working_set: &StateWorkingSet,
|
|
|
|
call: &Call,
|
|
|
|
input: PipelineData,
|
|
|
|
) -> Result<PipelineData, ShellError> {
|
|
|
|
let decl = working_set.get_decl(call.decl_id);
|
|
|
|
|
|
|
|
if !decl.is_const() {
|
|
|
|
return Err(ShellError::NotAConstCommand(call.head));
|
|
|
|
}
|
|
|
|
|
|
|
|
if !decl.is_known_external() && call.named_iter().any(|(flag, _, _)| flag.item == "help") {
|
|
|
|
// It would require re-implementing get_full_help() for const evaluation. Assuming that
|
|
|
|
// getting help messages at parse-time is rare enough, we can simply disallow it.
|
|
|
|
return Err(ShellError::NotAConstHelp(call.head));
|
|
|
|
}
|
|
|
|
|
|
|
|
decl.run_const(working_set, call, input)
|
|
|
|
}
|
|
|
|
|
2023-09-12 18:35:47 +00:00
|
|
|
pub fn eval_const_subexpression(
|
2023-08-26 13:41:29 +00:00
|
|
|
working_set: &StateWorkingSet,
|
|
|
|
block: &Block,
|
|
|
|
mut input: PipelineData,
|
2023-09-12 18:35:47 +00:00
|
|
|
span: Span,
|
2023-08-26 13:41:29 +00:00
|
|
|
) -> Result<PipelineData, ShellError> {
|
|
|
|
for pipeline in block.pipelines.iter() {
|
|
|
|
for element in pipeline.elements.iter() {
|
|
|
|
let PipelineElement::Expression(_, expr) = element else {
|
2023-09-12 18:35:47 +00:00
|
|
|
return Err(ShellError::NotAConstant(span));
|
2023-08-26 13:41:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
input = eval_constant_with_input(working_set, expr, input)?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(input)
|
|
|
|
}
|
|
|
|
|
2023-09-12 18:35:47 +00:00
|
|
|
pub fn eval_constant_with_input(
|
2023-08-26 13:41:29 +00:00
|
|
|
working_set: &StateWorkingSet,
|
|
|
|
expr: &Expression,
|
|
|
|
input: PipelineData,
|
|
|
|
) -> Result<PipelineData, ShellError> {
|
|
|
|
match &expr.expr {
|
|
|
|
Expr::Call(call) => eval_const_call(working_set, call, input),
|
|
|
|
Expr::Subexpression(block_id) => {
|
|
|
|
let block = working_set.get_block(*block_id);
|
2023-09-12 18:35:47 +00:00
|
|
|
eval_const_subexpression(working_set, block, input, expr.span)
|
2023-08-26 13:41:29 +00:00
|
|
|
}
|
|
|
|
_ => eval_constant(working_set, expr).map(|v| PipelineData::Value(v, None)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-21 22:21:03 +00:00
|
|
|
/// Evaluate a constant value at parse time
|
|
|
|
///
|
|
|
|
/// Based off eval_expression() in the engine
|
|
|
|
pub fn eval_constant(
|
|
|
|
working_set: &StateWorkingSet,
|
|
|
|
expr: &Expression,
|
2023-08-26 13:41:29 +00:00
|
|
|
) -> Result<Value, ShellError> {
|
2022-12-21 22:21:03 +00:00
|
|
|
match &expr.expr {
|
2023-07-21 13:20:33 +00:00
|
|
|
Expr::Bool(b) => Ok(Value::bool(*b, expr.span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
Expr::Int(i) => Ok(Value::int(*i, expr.span)),
|
|
|
|
Expr::Float(f) => Ok(Value::float(*f, expr.span)),
|
2023-09-03 14:27:29 +00:00
|
|
|
Expr::Binary(b) => Ok(Value::binary(b.clone(), expr.span)),
|
|
|
|
Expr::Filepath(path) => Ok(Value::string(path.clone(), expr.span)),
|
2023-07-13 21:02:05 +00:00
|
|
|
Expr::Var(var_id) => match working_set.get_variable(*var_id).const_val.as_ref() {
|
2022-12-21 22:21:03 +00:00
|
|
|
Some(val) => Ok(val.clone()),
|
2023-08-26 13:41:29 +00:00
|
|
|
None => Err(ShellError::NotAConstant(expr.span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
},
|
2023-09-03 14:27:29 +00:00
|
|
|
Expr::CellPath(cell_path) => Ok(Value::cell_path(cell_path.clone(), expr.span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
Expr::FullCellPath(cell_path) => {
|
|
|
|
let value = eval_constant(working_set, &cell_path.head)?;
|
|
|
|
|
2023-03-16 03:50:58 +00:00
|
|
|
match value.follow_cell_path(&cell_path.tail, false) {
|
2022-12-21 22:21:03 +00:00
|
|
|
Ok(val) => Ok(val),
|
|
|
|
// TODO: Better error conversion
|
2023-08-26 13:41:29 +00:00
|
|
|
Err(shell_error) => Err(ShellError::GenericError(
|
2022-12-21 22:21:03 +00:00
|
|
|
"Error when following cell path".to_string(),
|
2023-01-30 01:37:54 +00:00
|
|
|
format!("{shell_error:?}"),
|
2023-08-26 13:41:29 +00:00
|
|
|
Some(expr.span),
|
|
|
|
None,
|
|
|
|
vec![],
|
2022-12-21 22:21:03 +00:00
|
|
|
)),
|
|
|
|
}
|
|
|
|
}
|
2023-09-03 14:27:29 +00:00
|
|
|
Expr::DateTime(dt) => Ok(Value::date(*dt, expr.span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
Expr::List(x) => {
|
|
|
|
let mut output = vec![];
|
|
|
|
for expr in x {
|
|
|
|
output.push(eval_constant(working_set, expr)?);
|
|
|
|
}
|
2023-09-03 14:27:29 +00:00
|
|
|
Ok(Value::list(output, expr.span))
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
Expr::Record(fields) => {
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
let mut record = Record::new();
|
2022-12-21 22:21:03 +00:00
|
|
|
for (col, val) in fields {
|
|
|
|
// avoid duplicate cols.
|
|
|
|
let col_name = value_as_string(eval_constant(working_set, col)?, expr.span)?;
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
let pos = record.cols.iter().position(|c| c == &col_name);
|
2022-12-21 22:21:03 +00:00
|
|
|
match pos {
|
|
|
|
Some(index) => {
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
record.vals[index] = eval_constant(working_set, val)?;
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
None => {
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
record.push(col_name, eval_constant(working_set, val)?);
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
Ok(Value::record(record, expr.span))
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
Expr::Table(headers, vals) => {
|
|
|
|
let mut output_headers = vec![];
|
|
|
|
for expr in headers {
|
|
|
|
output_headers.push(value_as_string(
|
|
|
|
eval_constant(working_set, expr)?,
|
|
|
|
expr.span,
|
|
|
|
)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut output_rows = vec![];
|
|
|
|
for val in vals {
|
|
|
|
let mut row = vec![];
|
|
|
|
for expr in val {
|
|
|
|
row.push(eval_constant(working_set, expr)?);
|
|
|
|
}
|
Create `Record` type (#10103)
# Description
This PR creates a new `Record` type to reduce duplicate code and
possibly bugs as well. (This is an edited version of #9648.)
- `Record` implements `FromIterator` and `IntoIterator` and so can be
iterated over or collected into. For example, this helps with
conversions to and from (hash)maps. (Also, no more
`cols.iter().zip(vals)`!)
- `Record` has a `push(col, val)` function to help insure that the
number of columns is equal to the number of values. I caught a few
potential bugs thanks to this (e.g. in the `ls` command).
- Finally, this PR also adds a `record!` macro that helps simplify
record creation. It is used like so:
```rust
record! {
"key1" => some_value,
"key2" => Value::string("text", span),
"key3" => Value::int(optional_int.unwrap_or(0), span),
"key4" => Value::bool(config.setting, span),
}
```
Since macros hinder formatting, etc., the right hand side values should
be relatively short and sweet like the examples above.
Where possible, prefer `record!` or `.collect()` on an iterator instead
of multiple `Record::push`s, since the first two automatically set the
record capacity and do less work overall.
# User-Facing Changes
Besides the changes in `nu-protocol` the only other breaking changes are
to `nu-table::{ExpandedTable::build_map, JustTable::kv_table}`.
2023-08-24 19:50:29 +00:00
|
|
|
output_rows.push(Value::record(
|
|
|
|
Record {
|
|
|
|
cols: output_headers.clone(),
|
|
|
|
vals: row,
|
|
|
|
},
|
|
|
|
expr.span,
|
|
|
|
));
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
2023-09-03 14:27:29 +00:00
|
|
|
Ok(Value::list(output_rows, expr.span))
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
Expr::Keyword(_, _, expr) => eval_constant(working_set, expr),
|
2023-09-03 14:27:29 +00:00
|
|
|
Expr::String(s) => Ok(Value::string(s.clone(), expr.span)),
|
|
|
|
Expr::Nothing => Ok(Value::nothing(expr.span)),
|
2023-05-20 13:23:25 +00:00
|
|
|
Expr::ValueWithUnit(expr, unit) => {
|
|
|
|
if let Ok(Value::Int { val, .. }) = eval_constant(working_set, expr) {
|
2023-08-26 13:41:29 +00:00
|
|
|
unit.item.to_value(val, unit.span)
|
2023-05-20 13:23:25 +00:00
|
|
|
} else {
|
2023-08-26 13:41:29 +00:00
|
|
|
Err(ShellError::NotAConstant(expr.span))
|
2023-05-20 13:23:25 +00:00
|
|
|
}
|
|
|
|
}
|
2023-08-26 13:41:29 +00:00
|
|
|
Expr::Call(call) => {
|
|
|
|
Ok(eval_const_call(working_set, call, PipelineData::empty())?.into_value(expr.span))
|
|
|
|
}
|
|
|
|
Expr::Subexpression(block_id) => {
|
|
|
|
let block = working_set.get_block(*block_id);
|
|
|
|
Ok(
|
2023-09-12 18:35:47 +00:00
|
|
|
eval_const_subexpression(working_set, block, PipelineData::empty(), expr.span)?
|
2023-08-26 13:41:29 +00:00
|
|
|
.into_value(expr.span),
|
|
|
|
)
|
|
|
|
}
|
2023-09-05 14:35:58 +00:00
|
|
|
Expr::Range(from, next, to, operator) => {
|
|
|
|
let from = if let Some(f) = from {
|
|
|
|
eval_constant(working_set, f)?
|
|
|
|
} else {
|
|
|
|
Value::Nothing {
|
|
|
|
internal_span: expr.span,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let next = if let Some(s) = next {
|
|
|
|
eval_constant(working_set, s)?
|
|
|
|
} else {
|
|
|
|
Value::Nothing {
|
|
|
|
internal_span: expr.span,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let to = if let Some(t) = to {
|
|
|
|
eval_constant(working_set, t)?
|
|
|
|
} else {
|
|
|
|
Value::Nothing {
|
|
|
|
internal_span: expr.span,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Ok(Value::Range {
|
|
|
|
val: Box::new(Range::new(expr.span, from, next, to, operator)?),
|
|
|
|
internal_span: expr.span,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
Expr::UnaryNot(expr) => {
|
|
|
|
let lhs = eval_constant(working_set, expr)?;
|
|
|
|
match lhs {
|
|
|
|
Value::Bool { val, .. } => Ok(Value::bool(!val, expr.span)),
|
|
|
|
_ => Err(ShellError::TypeMismatch {
|
|
|
|
err_message: "bool".to_string(),
|
|
|
|
span: expr.span,
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Expr::BinaryOp(lhs, op, rhs) => {
|
|
|
|
let op_span = op.span;
|
|
|
|
let op = eval_operator(op)?;
|
|
|
|
|
|
|
|
match op {
|
|
|
|
Operator::Boolean(boolean) => {
|
|
|
|
let lhs = eval_constant(working_set, lhs)?;
|
|
|
|
match boolean {
|
|
|
|
Boolean::And => {
|
|
|
|
if lhs.is_false() {
|
|
|
|
Ok(Value::bool(false, expr.span))
|
|
|
|
} else {
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
lhs.and(op_span, &rhs, expr.span)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Boolean::Or => {
|
|
|
|
if lhs.is_true() {
|
|
|
|
Ok(Value::bool(true, expr.span))
|
|
|
|
} else {
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
lhs.or(op_span, &rhs, expr.span)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Boolean::Xor => {
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
lhs.xor(op_span, &rhs, expr.span)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Operator::Math(math) => {
|
|
|
|
let lhs = eval_constant(working_set, lhs)?;
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
|
|
|
|
match math {
|
|
|
|
Math::Plus => lhs.add(op_span, &rhs, expr.span),
|
|
|
|
Math::Minus => lhs.sub(op_span, &rhs, expr.span),
|
|
|
|
Math::Multiply => lhs.mul(op_span, &rhs, expr.span),
|
|
|
|
Math::Divide => lhs.div(op_span, &rhs, expr.span),
|
|
|
|
Math::Append => lhs.append(op_span, &rhs, expr.span),
|
|
|
|
Math::Modulo => lhs.modulo(op_span, &rhs, expr.span),
|
|
|
|
Math::FloorDivision => lhs.floor_div(op_span, &rhs, expr.span),
|
|
|
|
Math::Pow => lhs.pow(op_span, &rhs, expr.span),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Operator::Comparison(comparison) => {
|
|
|
|
let lhs = eval_constant(working_set, lhs)?;
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
match comparison {
|
|
|
|
Comparison::LessThan => lhs.lt(op_span, &rhs, expr.span),
|
|
|
|
Comparison::LessThanOrEqual => lhs.lte(op_span, &rhs, expr.span),
|
|
|
|
Comparison::GreaterThan => lhs.gt(op_span, &rhs, expr.span),
|
|
|
|
Comparison::GreaterThanOrEqual => lhs.gte(op_span, &rhs, expr.span),
|
|
|
|
Comparison::Equal => lhs.eq(op_span, &rhs, expr.span),
|
|
|
|
Comparison::NotEqual => lhs.ne(op_span, &rhs, expr.span),
|
|
|
|
Comparison::In => lhs.r#in(op_span, &rhs, expr.span),
|
|
|
|
Comparison::NotIn => lhs.not_in(op_span, &rhs, expr.span),
|
|
|
|
Comparison::StartsWith => lhs.starts_with(op_span, &rhs, expr.span),
|
|
|
|
Comparison::EndsWith => lhs.ends_with(op_span, &rhs, expr.span),
|
|
|
|
// RegEx comparison is not a constant
|
|
|
|
_ => Err(ShellError::NotAConstant(expr.span)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Operator::Bits(bits) => {
|
|
|
|
let lhs = eval_constant(working_set, lhs)?;
|
|
|
|
let rhs = eval_constant(working_set, rhs)?;
|
|
|
|
match bits {
|
|
|
|
Bits::BitAnd => lhs.bit_and(op_span, &rhs, expr.span),
|
|
|
|
Bits::BitOr => lhs.bit_or(op_span, &rhs, expr.span),
|
|
|
|
Bits::BitXor => lhs.bit_xor(op_span, &rhs, expr.span),
|
|
|
|
Bits::ShiftLeft => lhs.bit_shl(op_span, &rhs, expr.span),
|
|
|
|
Bits::ShiftRight => lhs.bit_shr(op_span, &rhs, expr.span),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Operator::Assignment(_) => Err(ShellError::NotAConstant(expr.span)),
|
|
|
|
}
|
|
|
|
}
|
2023-09-12 18:35:47 +00:00
|
|
|
Expr::Block(block_id) => Ok(Value::block(*block_id, expr.span)),
|
2023-08-26 13:41:29 +00:00
|
|
|
_ => Err(ShellError::NotAConstant(expr.span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the value as a string
|
2023-08-26 13:41:29 +00:00
|
|
|
pub fn value_as_string(value: Value, span: Span) -> Result<String, ShellError> {
|
2022-12-21 22:21:03 +00:00
|
|
|
match value {
|
|
|
|
Value::String { val, .. } => Ok(val),
|
2023-08-26 13:41:29 +00:00
|
|
|
_ => Err(ShellError::NotAConstant(span)),
|
2022-12-21 22:21:03 +00:00
|
|
|
}
|
|
|
|
}
|