Friendly error message for access beyond end (#6944)

Adds `ShellError::AccessEmptyContent`
This commit is contained in:
WindSoilder 2022-10-30 01:47:50 +08:00 committed by GitHub
parent 7039602e4d
commit 4f7f6a2932
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 46 additions and 11 deletions

View file

@ -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()

View file

@ -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));
} }

View file

@ -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!(

View file

@ -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,
)) ))
} }

View file

@ -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,
)) ))
} }

View file

@ -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

View file

@ -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()?)),