mirror of
https://github.com/nushell/nushell
synced 2025-01-13 13:49:21 +00:00
Friendly error message for access beyond end (#6944)
Adds `ShellError::AccessEmptyContent`
This commit is contained in:
parent
7039602e4d
commit
4f7f6a2932
7 changed files with 46 additions and 11 deletions
|
@ -316,7 +316,7 @@ impl NuDataFrame {
|
||||||
let column = conversion::create_column(&series, row, row + 1, span)?;
|
let column = conversion::create_column(&series, row, row + 1, span)?;
|
||||||
|
|
||||||
if column.len() == 0 {
|
if column.len() == 0 {
|
||||||
Err(ShellError::AccessBeyondEnd(series.len(), span))
|
Err(ShellError::AccessEmptyContent(span))
|
||||||
} else {
|
} else {
|
||||||
let value = column
|
let value = column
|
||||||
.into_iter()
|
.into_iter()
|
||||||
|
|
|
@ -138,6 +138,8 @@ fn update(
|
||||||
for idx in 0..*val {
|
for idx in 0..*val {
|
||||||
if let Some(v) = input.next() {
|
if let Some(v) = input.next() {
|
||||||
pre_elems.push(v);
|
pre_elems.push(v);
|
||||||
|
} else if idx == 0 {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*span));
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(idx - 1, *span));
|
return Err(ShellError::AccessBeyondEnd(idx - 1, *span));
|
||||||
}
|
}
|
||||||
|
|
|
@ -199,11 +199,17 @@ fn errors_fetching_by_index_out_of_bounds() {
|
||||||
"#
|
"#
|
||||||
));
|
));
|
||||||
|
|
||||||
assert!(actual.err.contains("Row number too large (max: 3)"),);
|
assert!(actual.err.contains("Row number too large (max: 2)"),);
|
||||||
assert!(actual.err.contains("too large"),);
|
assert!(actual.err.contains("too large"),);
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn errors_fetching_by_accessing_empty_list() {
|
||||||
|
let actual = nu!(cwd: ".", pipeline(r#"[] | get 3"#));
|
||||||
|
assert!(actual.err.contains("Row number too large (empty content)"),);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn quoted_column_access() {
|
fn quoted_column_access() {
|
||||||
let actual = nu!(
|
let actual = nu!(
|
||||||
|
|
|
@ -90,9 +90,11 @@ impl CallExt for Call {
|
||||||
if let Some(expr) = self.positional_nth(pos) {
|
if let Some(expr) = self.positional_nth(pos) {
|
||||||
let result = eval_expression(engine_state, stack, expr)?;
|
let result = eval_expression(engine_state, stack, expr)?;
|
||||||
FromValue::from_value(&result)
|
FromValue::from_value(&result)
|
||||||
|
} else if self.positional_len() == 0 {
|
||||||
|
Err(ShellError::AccessEmptyContent(self.head))
|
||||||
} else {
|
} else {
|
||||||
Err(ShellError::AccessBeyondEnd(
|
Err(ShellError::AccessBeyondEnd(
|
||||||
self.positional_len(),
|
self.positional_len() - 1,
|
||||||
self.head,
|
self.head,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
|
@ -96,9 +96,11 @@ impl EvaluatedCall {
|
||||||
pub fn req<T: FromValue>(&self, pos: usize) -> Result<T, ShellError> {
|
pub fn req<T: FromValue>(&self, pos: usize) -> Result<T, ShellError> {
|
||||||
if let Some(value) = self.nth(pos) {
|
if let Some(value) = self.nth(pos) {
|
||||||
FromValue::from_value(&value)
|
FromValue::from_value(&value)
|
||||||
|
} else if self.positional.is_empty() {
|
||||||
|
Err(ShellError::AccessEmptyContent(self.head))
|
||||||
} else {
|
} else {
|
||||||
Err(ShellError::AccessBeyondEnd(
|
Err(ShellError::AccessBeyondEnd(
|
||||||
self.positional.len(),
|
self.positional.len() - 1,
|
||||||
self.head,
|
self.head,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
|
@ -358,6 +358,15 @@ Either make sure {0} is a string, or add a 'to_string' entry for it in ENV_CONVE
|
||||||
#[diagnostic(code(nu::shell::access_beyond_end), url(docsrs))]
|
#[diagnostic(code(nu::shell::access_beyond_end), url(docsrs))]
|
||||||
AccessBeyondEnd(usize, #[label = "index too large (max: {0})"] Span),
|
AccessBeyondEnd(usize, #[label = "index too large (max: {0})"] Span),
|
||||||
|
|
||||||
|
/// You attempted to access an index when it's empty.
|
||||||
|
///
|
||||||
|
/// ## Resolution
|
||||||
|
///
|
||||||
|
/// Check your lengths and try again.
|
||||||
|
#[error("Row number too large (empty content).")]
|
||||||
|
#[diagnostic(code(nu::shell::access_beyond_end), url(docsrs))]
|
||||||
|
AccessEmptyContent(#[label = "index too large (empty content)"] Span),
|
||||||
|
|
||||||
/// You attempted to access an index beyond the available length of a stream.
|
/// You attempted to access an index beyond the available length of a stream.
|
||||||
///
|
///
|
||||||
/// ## Resolution
|
/// ## Resolution
|
||||||
|
|
|
@ -641,8 +641,10 @@ impl Value {
|
||||||
Value::List { vals: val, .. } => {
|
Value::List { vals: val, .. } => {
|
||||||
if let Some(item) = val.get(*count) {
|
if let Some(item) = val.get(*count) {
|
||||||
current = item.clone();
|
current = item.clone();
|
||||||
|
} else if val.is_empty() {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*origin_span))
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(val.len(), *origin_span));
|
return Err(ShellError::AccessBeyondEnd(val.len() - 1, *origin_span));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Value::Binary { val, .. } => {
|
Value::Binary { val, .. } => {
|
||||||
|
@ -651,8 +653,10 @@ impl Value {
|
||||||
val: *item as i64,
|
val: *item as i64,
|
||||||
span: *origin_span,
|
span: *origin_span,
|
||||||
};
|
};
|
||||||
|
} else if val.is_empty() {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*origin_span))
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(val.len(), *origin_span));
|
return Err(ShellError::AccessBeyondEnd(val.len() - 1, *origin_span));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Value::Range { val, .. } => {
|
Value::Range { val, .. } => {
|
||||||
|
@ -839,8 +843,10 @@ impl Value {
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
if let Some(v) = vals.get_mut(*row_num) {
|
if let Some(v) = vals.get_mut(*row_num) {
|
||||||
v.upsert_data_at_cell_path(&cell_path[1..], new_val)?
|
v.upsert_data_at_cell_path(&cell_path[1..], new_val)?
|
||||||
|
} else if vals.is_empty() {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*span));
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(vals.len(), *span));
|
return Err(ShellError::AccessBeyondEnd(vals.len() - 1, *span));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
||||||
|
@ -931,8 +937,10 @@ impl Value {
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
if let Some(v) = vals.get_mut(*row_num) {
|
if let Some(v) = vals.get_mut(*row_num) {
|
||||||
v.update_data_at_cell_path(&cell_path[1..], new_val)?
|
v.update_data_at_cell_path(&cell_path[1..], new_val)?
|
||||||
|
} else if vals.is_empty() {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*span));
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(vals.len(), *span));
|
return Err(ShellError::AccessBeyondEnd(vals.len() - 1, *span));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
||||||
|
@ -1005,8 +1013,10 @@ impl Value {
|
||||||
if vals.get_mut(*row_num).is_some() {
|
if vals.get_mut(*row_num).is_some() {
|
||||||
vals.remove(*row_num);
|
vals.remove(*row_num);
|
||||||
Ok(())
|
Ok(())
|
||||||
|
} else if vals.is_empty() {
|
||||||
|
Err(ShellError::AccessEmptyContent(*span))
|
||||||
} else {
|
} else {
|
||||||
Err(ShellError::AccessBeyondEnd(vals.len(), *span))
|
Err(ShellError::AccessBeyondEnd(vals.len() - 1, *span))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v => Err(ShellError::NotAList(*span, v.span()?)),
|
v => Err(ShellError::NotAList(*span, v.span()?)),
|
||||||
|
@ -1069,8 +1079,10 @@ impl Value {
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
if let Some(v) = vals.get_mut(*row_num) {
|
if let Some(v) = vals.get_mut(*row_num) {
|
||||||
v.remove_data_at_cell_path(&cell_path[1..])
|
v.remove_data_at_cell_path(&cell_path[1..])
|
||||||
|
} else if vals.is_empty() {
|
||||||
|
Err(ShellError::AccessEmptyContent(*span))
|
||||||
} else {
|
} else {
|
||||||
Err(ShellError::AccessBeyondEnd(vals.len(), *span))
|
Err(ShellError::AccessBeyondEnd(vals.len() - 1, *span))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v => Err(ShellError::NotAList(*span, v.span()?)),
|
v => Err(ShellError::NotAList(*span, v.span()?)),
|
||||||
|
@ -1157,8 +1169,10 @@ impl Value {
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
if let Some(v) = vals.get_mut(*row_num) {
|
if let Some(v) = vals.get_mut(*row_num) {
|
||||||
v.insert_data_at_cell_path(&cell_path[1..], new_val)?
|
v.insert_data_at_cell_path(&cell_path[1..], new_val)?
|
||||||
|
} else if vals.is_empty() {
|
||||||
|
return Err(ShellError::AccessEmptyContent(*span));
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::AccessBeyondEnd(vals.len(), *span));
|
return Err(ShellError::AccessBeyondEnd(vals.len() - 1, *span));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
v => return Err(ShellError::NotAList(*span, v.span()?)),
|
||||||
|
|
Loading…
Reference in a new issue