From b596c73eb050118f9f9e114dad41018b07bf12d6 Mon Sep 17 00:00:00 2001 From: Lovecraftian Horror Date: Sat, 18 Dec 2021 16:21:50 -0600 Subject: [PATCH 1/2] Fix panic from slice end being before slice start --- Cargo.lock | 78 ++++++++++++++++++++++++++++++++++++++++++++++++ Cargo.toml | 3 +- src/utils/ext.rs | 19 +++++++++++- 3 files changed, 98 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index c909f1074..fa0c3ba7e 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -55,6 +55,21 @@ version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cdb031dd78e28731d87d56cc8ffef4a8f36ca26c38fe2de700543e627f8a464a" +[[package]] +name = "bit-set" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e11e16035ea35e4e5997b393eacbf6f63983188f7a2ad25bfb13465f5ad59de" +dependencies = [ + "bit-vec", +] + +[[package]] +name = "bit-vec" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "349f9b6a179ed607305526ca489b34ad0a41aed5f7980fa90eb03160b69598fb" + [[package]] name = "bitflags" version = "1.2.1" @@ -766,6 +781,38 @@ dependencies = [ "unicode-xid", ] +[[package]] +name = "proptest" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e0d9cc07f18492d879586c92b485def06bc850da3118075cd45d50e9c95b0e5" +dependencies = [ + "bit-set", + "bitflags", + "byteorder", + "lazy_static", + "num-traits", + "quick-error 2.0.1", + "rand", + "rand_chacha", + "rand_xorshift", + "regex-syntax", + "rusty-fork", + "tempfile", +] + +[[package]] +name = "quick-error" +version = "1.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0" + +[[package]] +name = "quick-error" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a993555f31e5a609f617c12db6250dedcac1b0a85076912c436e6fc9b2c8e6a3" + [[package]] name = "quote" version = "1.0.9" @@ -815,6 +862,15 @@ dependencies = [ "rand_core", ] +[[package]] +name = "rand_xorshift" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d25bf25ec5ae4a3f1b92f929810509a2f53d7dca2f50b794ff57e3face536c8f" +dependencies = [ + "rand_core", +] + [[package]] name = "rayon" version = "1.5.1" @@ -885,6 +941,18 @@ dependencies = [ "winapi", ] +[[package]] +name = "rusty-fork" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb3dcc6e454c328bb824492db107ab7c0ae8fcffe4ad210136ef014458c1bc4f" +dependencies = [ + "fnv", + "quick-error 1.2.3", + "tempfile", + "wait-timeout", +] + [[package]] name = "ryu" version = "1.0.5" @@ -1123,6 +1191,7 @@ dependencies = [ "num-format", "once_cell", "parking_lot", + "proptest", "rayon", "regex", "serde", @@ -1260,6 +1329,15 @@ version = "0.8.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f1bddf1187be692e79c5ffeab891132dfb0f236ed36a43c7ed39f1165ee20191" +[[package]] +name = "wait-timeout" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f200f5b12eb75f8c1ed65abd4b2db8a6e1b138a20de009dacee265a2498f3f6" +dependencies = [ + "libc", +] + [[package]] name = "walkdir" version = "2.3.2" diff --git a/Cargo.toml b/Cargo.toml index a4957d164..5ba232325 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -75,6 +75,7 @@ optional = true version = "0.8.15" [dev-dependencies] +proptest = "1.0.0" regex = "1.4.6" tempfile = "3.0.8" -git2 = { version="0.13.20", default-features=false, features=[] } \ No newline at end of file +git2 = { version="0.13.20", default-features=false, features=[] } diff --git a/src/utils/ext.rs b/src/utils/ext.rs index 5169fd51b..3511840f8 100644 --- a/src/utils/ext.rs +++ b/src/utils/ext.rs @@ -29,6 +29,7 @@ pub(crate) trait SliceExt { impl SliceExt for [u8] { fn trim_first_and_last_line_of_whitespace(&self) -> &Self { + dbg!(&self); let start = self .iter() .position(|c| c.is_line_ending_whitespace() || !c.is_whitespace()) @@ -40,7 +41,8 @@ impl SliceExt for [u8] { .map_or_else( || self.len(), |i| { - if self[i.saturating_sub(1)] == b'\r' { + // Remove the entire `\r\n` in the case that it was the line ending whitespace + if self[i.saturating_sub(1)] == b'\r' && self[i] == b'\n' { i - 1 } else { i @@ -114,6 +116,8 @@ impl SliceExt for [u8] { mod tests { use super::*; + use proptest::prelude::*; + #[test] fn is_whitespace() { assert!(b' '.is_whitespace()); @@ -142,4 +146,17 @@ mod tests { assert!([1, 2, 3, 4, 5].contains_slice(&[2, 3, 4])); assert!(![1, 2, 3, 4, 5].contains_slice(&[])); } + + #[test] + fn issue_727() { + assert_eq!(b"", b"\ra ".trim_first_and_last_line_of_whitespace()); + assert_eq!(b"a", b"\r\na ".trim_first_and_last_line_of_whitespace()); + } + + proptest! { + #[test] + fn trim_whitespace_doesnt_panic(input: Vec) { + let _ = &input.trim_first_and_last_line_of_whitespace(); + } + } } From 1f3dd491b3c578c026148d560d2999c427e5a694 Mon Sep 17 00:00:00 2001 From: Lovecraftian Horror Date: Sat, 18 Dec 2021 16:31:15 -0600 Subject: [PATCH 2/2] Fix fallback value for end index --- src/utils/ext.rs | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/utils/ext.rs b/src/utils/ext.rs index 3511840f8..c9cd516dd 100644 --- a/src/utils/ext.rs +++ b/src/utils/ext.rs @@ -29,7 +29,6 @@ pub(crate) trait SliceExt { impl SliceExt for [u8] { fn trim_first_and_last_line_of_whitespace(&self) -> &Self { - dbg!(&self); let start = self .iter() .position(|c| c.is_line_ending_whitespace() || !c.is_whitespace()) @@ -39,7 +38,7 @@ impl SliceExt for [u8] { .iter() .rposition(|c| c.is_line_ending_whitespace() || !c.is_whitespace()) .map_or_else( - || self.len(), + || self.len().saturating_sub(1), |i| { // Remove the entire `\r\n` in the case that it was the line ending whitespace if self[i.saturating_sub(1)] == b'\r' && self[i] == b'\n' { @@ -148,14 +147,16 @@ mod tests { } #[test] - fn issue_727() { + fn trim_first_and_last_line_of_whitespace_edge_cases() { assert_eq!(b"", b"\ra ".trim_first_and_last_line_of_whitespace()); assert_eq!(b"a", b"\r\na ".trim_first_and_last_line_of_whitespace()); + + assert_eq!(b" ", b" ".trim_first_and_last_line_of_whitespace()); } proptest! { #[test] - fn trim_whitespace_doesnt_panic(input: Vec) { + fn trim_first_and_last_line_of_whitespace_doesnt_panic(input: Vec) { let _ = &input.trim_first_and_last_line_of_whitespace(); } }