Skip to content

Commit 4682271

Browse files
committed
Auto merge of #38066 - bluss:string-slice-error, r=sfackler
Use more specific panic message for &str slicing errors Separate out of bounds errors from character boundary errors, and print more details for character boundary errors. It reports the first error it finds in: 1. begin out of bounds 2. end out of bounds 3. begin <= end violated 3. begin not char boundary 5. end not char boundary. Example: &"abcαβγ"[..4] thread 'str::test_slice_fail_boundary_1' panicked at 'byte index 4 is not a char boundary; it is inside 'α' (bytes 3..5) of `abcαβγ`' Fixes #38052
2 parents 8f62c29 + d83fff3 commit 4682271

File tree

3 files changed

+38
-8
lines changed

3 files changed

+38
-8
lines changed

src/doc/book/strings.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -163,8 +163,8 @@ let hachi = &dog[0..2];
163163
with this error:
164164

165165
```text
166-
thread 'main' panicked at 'index 0 and/or 2 in `忠犬ハチ公` do not lie on
167-
character boundary'
166+
thread 'main' panicked at 'byte index 2 is not a char boundary; it is inside '忠'
167+
(bytes 0..3) of `忠犬ハチ公`'
168168
```
169169

170170
## Concatenation

src/libcollectionstest/str.rs

+14-2
Original file line numberDiff line numberDiff line change
@@ -383,17 +383,29 @@ tempus vel, gravida nec quam.";
383383

384384
// check the panic includes the prefix of the sliced string
385385
#[test]
386-
#[should_panic(expected="Lorem ipsum dolor sit amet")]
386+
#[should_panic(expected="byte index 1024 is out of bounds of `Lorem ipsum dolor sit amet")]
387387
fn test_slice_fail_truncated_1() {
388388
&LOREM_PARAGRAPH[..1024];
389389
}
390390
// check the truncation in the panic message
391391
#[test]
392-
#[should_panic(expected="luctus, im`[...] do not lie on character boundary")]
392+
#[should_panic(expected="luctus, im`[...]")]
393393
fn test_slice_fail_truncated_2() {
394394
&LOREM_PARAGRAPH[..1024];
395395
}
396396

397+
#[test]
398+
#[should_panic(expected="byte index 4 is not a char boundary; it is inside 'α' (bytes 3..5) of")]
399+
fn test_slice_fail_boundary_1() {
400+
&"abcαβγ"[4..];
401+
}
402+
403+
#[test]
404+
#[should_panic(expected="byte index 6 is not a char boundary; it is inside 'β' (bytes 5..7) of")]
405+
fn test_slice_fail_boundary_2() {
406+
&"abcαβγ"[2..6];
407+
}
408+
397409
#[test]
398410
fn test_slice_from() {
399411
assert_eq!(&"abcd"[0..], "abcd");

src/libcore/str/mod.rs

+22-4
Original file line numberDiff line numberDiff line change
@@ -1746,13 +1746,31 @@ fn truncate_to_char_boundary(s: &str, mut max: usize) -> (bool, &str) {
17461746
#[cold]
17471747
fn slice_error_fail(s: &str, begin: usize, end: usize) -> ! {
17481748
const MAX_DISPLAY_LENGTH: usize = 256;
1749-
let (truncated, s) = truncate_to_char_boundary(s, MAX_DISPLAY_LENGTH);
1749+
let (truncated, s_trunc) = truncate_to_char_boundary(s, MAX_DISPLAY_LENGTH);
17501750
let ellipsis = if truncated { "[...]" } else { "" };
17511751

1752+
// 1. out of bounds
1753+
if begin > s.len() || end > s.len() {
1754+
let oob_index = if begin > s.len() { begin } else { end };
1755+
panic!("byte index {} is out of bounds of `{}`{}", oob_index, s_trunc, ellipsis);
1756+
}
1757+
1758+
// 2. begin <= end
17521759
assert!(begin <= end, "begin <= end ({} <= {}) when slicing `{}`{}",
1753-
begin, end, s, ellipsis);
1754-
panic!("index {} and/or {} in `{}`{} do not lie on character boundary",
1755-
begin, end, s, ellipsis);
1760+
begin, end, s_trunc, ellipsis);
1761+
1762+
// 3. character boundary
1763+
let index = if !s.is_char_boundary(begin) { begin } else { end };
1764+
// find the character
1765+
let mut char_start = index;
1766+
while !s.is_char_boundary(char_start) {
1767+
char_start -= 1;
1768+
}
1769+
// `char_start` must be less than len and a char boundary
1770+
let ch = s[char_start..].chars().next().unwrap();
1771+
let char_range = char_start .. char_start + ch.len_utf8();
1772+
panic!("byte index {} is not a char boundary; it is inside {:?} (bytes {:?}) of `{}`{}",
1773+
index, ch, char_range, s_trunc, ellipsis);
17561774
}
17571775

17581776
#[stable(feature = "core", since = "1.6.0")]

0 commit comments

Comments
 (0)