Fix highlighting for slices with quotes

This would highlight `$var["foo"]` as an error because
parse_util_slice_length didn't advance the iterator.

There's got to be a nicer way to write this.
This commit is contained in:
Fabian Boehm 2024-01-10 17:04:54 +01:00
parent e84d110995
commit fbfc6f8d8c

View file

@ -31,7 +31,7 @@ use crate::wildcard::{ANY_CHAR, ANY_STRING, ANY_STRING_RECURSIVE};
use std::ops; use std::ops;
/// Handles slices: the square brackets in an expression like $foo[5..4] /// Handles slices: the square brackets in an expression like $foo[5..4]
/// \return the length of the slice starting at \p in, or 0 if there is no slice, or -1 on error. /// \return the length of the slice starting at \p in, or 0 if there is no slice, or None on error.
/// This never accepts incomplete slices. /// This never accepts incomplete slices.
pub fn parse_util_slice_length(input: &wstr) -> Option<usize> { pub fn parse_util_slice_length(input: &wstr) -> Option<usize> {
const openc: char = '['; const openc: char = '[';
@ -46,11 +46,20 @@ pub fn parse_util_slice_length(input: &wstr) -> Option<usize> {
let mut bracket_count = 1; let mut bracket_count = 1;
let mut pos = 0; let mut pos = 0;
for c in chars { while let Some(c) = chars.next() {
pos += 1; pos += 1;
if !escaped { if !escaped {
if ['\'', '"'].contains(&c) { if ['\'', '"'].contains(&c) {
let oldpos = pos;
pos = quote_end(input, pos, c)?; pos = quote_end(input, pos, c)?;
// We need to advance the iterator as well
if pos - oldpos > 0 {
// nth(0) advances by 1
chars.nth(pos - oldpos - 1)?;
} else {
// Quotes aren't over, slice is invalid
return None;
}
} else if c == openc { } else if c == openc {
bracket_count += 1; bracket_count += 1;
} else if c == closec { } else if c == closec {
@ -1804,6 +1813,16 @@ fn test_parse_util_cmdsubst_extent() {
); );
} }
#[test]
#[serial]
fn test_parse_util_slice_length() {
test_init();
assert_eq!(parse_util_slice_length(L!("[2]")), Some(3));
assert_eq!(parse_util_slice_length(L!("[12]")), Some(4));
assert_eq!(parse_util_slice_length(L!("[\"foo\"]")), Some(7));
assert_eq!(parse_util_slice_length(L!("[\"foo\"")), None);
}
#[test] #[test]
#[serial] #[serial]
fn test_escape_quotes() { fn test_escape_quotes() {