From 250ee62e16276a0864dacf0e2f698563a55fcb4a Mon Sep 17 00:00:00 2001 From: nibon7 Date: Thu, 30 Nov 2023 22:13:10 +0800 Subject: [PATCH] Add boundary check for str index-of (#11190) # Description This PR tries to fix #10774 # User-Facing Changes * before ![Screenshot from 2023-11-30 20-18-56](https://github.com/nushell/nushell/assets/15247421/4559e114-0757-4a73-91e7-c7536a8ce5f1) ![Screenshot from 2023-11-30 20-19-23](https://github.com/nushell/nushell/assets/15247421/30ab1b5a-3ec3-48a8-ae76-65721b650fcf) * after ![Screenshot from 2023-11-30 20-21-04](https://github.com/nushell/nushell/assets/15247421/f34fd276-dccf-4328-b9d2-bf368b5b3ae5) ![Screenshot from 2023-11-30 20-20-39](https://github.com/nushell/nushell/assets/15247421/653e47d8-8d68-4ef3-adef-0865179c4f9b) # Tests + Formatting # After Submitting --- .../nu-command/src/strings/str_/index_of.rs | 104 +++++++++++++++++- 1 file changed, 98 insertions(+), 6 deletions(-) diff --git a/crates/nu-command/src/strings/str_/index_of.rs b/crates/nu-command/src/strings/str_/index_of.rs index 6dfd83bcd2..9287ece986 100644 --- a/crates/nu-command/src/strings/str_/index_of.rs +++ b/crates/nu-command/src/strings/str_/index_of.rs @@ -13,7 +13,7 @@ use unicode_segmentation::UnicodeSegmentation; struct Arguments { end: bool, substring: String, - range: Option, + range: Option>, cell_paths: Option>, graphemes: bool, } @@ -147,7 +147,10 @@ fn action( ) -> Value { match input { Value::String { val: s, .. } => { - let (start_index, end_index) = if let Some(range) = range { + let mut range_span = head; + let (start_index, end_index) = if let Some(spanned_range) = range { + range_span = spanned_range.span; + let range = &spanned_range.item; match util::process_range(range) { Ok(r) => { // `process_range()` returns `isize::MAX` if the range is open-ended, @@ -168,6 +171,17 @@ fn action( (0usize, s.len()) }; + if s.get(start_index..end_index).is_none() { + return Value::error( + ShellError::InvalidRange { + left_flank: start_index.to_string(), + right_flank: end_index.to_string(), + span: range_span, + }, + head, + ); + } + // When the -e flag is present, search using rfind instead of find.s if let Some(result) = if *end { s[start_index..end_index].rfind(&**substring) @@ -265,10 +279,15 @@ mod tests { inclusion: RangeInclusion::Inclusive, }; + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + let options = Arguments { substring: String::from("Cargo"), - range: Some(range), + range: Some(spanned_range), cell_paths: None, end: false, graphemes: false, @@ -288,10 +307,15 @@ mod tests { to: Value::int(5, Span::test_data()), }; + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + let options = Arguments { substring: String::from("Banana"), - range: Some(range), + range: Some(spanned_range), cell_paths: None, end: false, graphemes: false, @@ -311,10 +335,15 @@ mod tests { inclusion: RangeInclusion::Inclusive, }; + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + let options = Arguments { substring: String::from("123"), - range: Some(range), + range: Some(spanned_range), cell_paths: None, end: false, graphemes: false, @@ -334,10 +363,15 @@ mod tests { inclusion: RangeInclusion::RightExclusive, }; + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + let options = Arguments { substring: String::from("1"), - range: Some(range), + range: Some(spanned_range), cell_paths: None, end: false, graphemes: false, @@ -363,4 +397,62 @@ mod tests { let actual = action(&word, &options, Span::test_data()); assert_eq!(actual, Value::test_int(15)); } + + #[test] + fn index_is_not_a_char_boundary() { + let word = Value::string(String::from("💛"), Span::test_data()); + + let range = Range { + from: Value::int(0, Span::test_data()), + incr: Value::int(1, Span::test_data()), + to: Value::int(3, Span::test_data()), + inclusion: RangeInclusion::Inclusive, + }; + + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + + let options = Arguments { + substring: String::new(), + + range: Some(spanned_range), + cell_paths: None, + end: false, + graphemes: false, + }; + + let actual = action(&word, &options, Span::test_data()); + assert!(actual.is_error()); + } + + #[test] + fn index_is_out_of_bounds() { + let word = Value::string(String::from("hello"), Span::test_data()); + + let range = Range { + from: Value::int(-1, Span::test_data()), + incr: Value::int(1, Span::test_data()), + to: Value::int(3, Span::test_data()), + inclusion: RangeInclusion::Inclusive, + }; + + let spanned_range = Spanned { + item: range, + span: Span::test_data(), + }; + + let options = Arguments { + substring: String::from("h"), + + range: Some(spanned_range), + cell_paths: None, + end: false, + graphemes: false, + }; + + let actual = action(&word, &options, Span::test_data()); + assert!(actual.is_error()); + } }