Bytes checked api (#288)

This commit is contained in:
Nikolay Kim 2024-02-01 20:28:17 +06:00 committed by GitHub
parent 18680ea05b
commit dc4bbf02c8
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 104 additions and 51 deletions

View file

@ -1,5 +1,9 @@
# Changes # Changes
## [0.1.24] (2024-02-01)
* Add `checked` api
## [0.1.23] (2024-01-19) ## [0.1.23] (2024-01-19)
* Revert changes to release shared_vec * Revert changes to release shared_vec

View file

@ -1,6 +1,6 @@
[package] [package]
name = "ntex-bytes" name = "ntex-bytes"
version = "0.1.23" version = "0.1.24"
authors = ["Nikolay Kim <fafhrd91@gmail.com>", "Carl Lerche <me@carllerche.com>"] authors = ["Nikolay Kim <fafhrd91@gmail.com>", "Carl Lerche <me@carllerche.com>"]
description = "Types and traits for working with bytes (bytes crate fork)" description = "Types and traits for working with bytes (bytes crate fork)"
documentation = "https://docs.rs/ntex-bytes" documentation = "https://docs.rs/ntex-bytes"

View file

@ -549,6 +549,14 @@ impl Bytes {
/// Requires that `begin <= end` and `end <= self.len()`, otherwise slicing /// Requires that `begin <= end` and `end <= self.len()`, otherwise slicing
/// will panic. /// will panic.
pub fn slice(&self, range: impl RangeBounds<usize>) -> Bytes { pub fn slice(&self, range: impl RangeBounds<usize>) -> Bytes {
self.slice_checked(range)
.expect("Requires that `begin <= end` and `end <= self.len()`")
}
/// Returns a slice of self for the provided range.
///
/// Does nothing if `begin <= end` or `end <= self.len()`
pub fn slice_checked(&self, range: impl RangeBounds<usize>) -> Option<Bytes> {
use std::ops::Bound; use std::ops::Bound;
let len = self.len(); let len = self.len();
@ -565,21 +573,21 @@ impl Bytes {
Bound::Unbounded => len, Bound::Unbounded => len,
}; };
assert!(begin <= end); if begin <= end && end <= len {
assert!(end <= len); if end - begin <= INLINE_CAP {
Some(Bytes {
if end - begin <= INLINE_CAP { inner: Inner::from_slice_inline(&self[begin..end]),
Bytes { })
inner: Inner::from_slice_inline(&self[begin..end]), } else {
let mut ret = self.clone();
unsafe {
ret.inner.set_end(end);
ret.inner.set_start(begin);
}
Some(ret)
} }
} else { } else {
let mut ret = self.clone(); None
unsafe {
ret.inner.set_end(end);
ret.inner.set_start(begin);
}
ret
} }
} }
@ -609,18 +617,24 @@ impl Bytes {
/// Requires that the given `sub` slice is in fact contained within the /// Requires that the given `sub` slice is in fact contained within the
/// `Bytes` buffer; otherwise this function will panic. /// `Bytes` buffer; otherwise this function will panic.
pub fn slice_ref(&self, subset: &[u8]) -> Bytes { pub fn slice_ref(&self, subset: &[u8]) -> Bytes {
self.slice_ref_checked(subset)
.expect("Given `sub` slice is not contained within the `Bytes` buffer")
}
/// Returns a slice of self that is equivalent to the given `subset`.
pub fn slice_ref_checked(&self, subset: &[u8]) -> Option<Bytes> {
let bytes_p = self.as_ptr() as usize; let bytes_p = self.as_ptr() as usize;
let bytes_len = self.len(); let bytes_len = self.len();
let sub_p = subset.as_ptr() as usize; let sub_p = subset.as_ptr() as usize;
let sub_len = subset.len(); let sub_len = subset.len();
assert!(sub_p >= bytes_p); if sub_p >= bytes_p && sub_p + sub_len <= bytes_p + bytes_len {
assert!(sub_p + sub_len <= bytes_p + bytes_len); let sub_offset = sub_p - bytes_p;
Some(self.slice(sub_offset..(sub_offset + sub_len)))
let sub_offset = sub_p - bytes_p; } else {
None
self.slice(sub_offset..(sub_offset + sub_len)) }
} }
/// Splits the bytes into two at the given index. /// Splits the bytes into two at the given index.
@ -645,20 +659,28 @@ impl Bytes {
/// ///
/// # Panics /// # Panics
/// ///
/// Panics if `at > len`. /// Panics if `at > self.len()`.
pub fn split_off(&mut self, at: usize) -> Bytes { pub fn split_off(&mut self, at: usize) -> Bytes {
assert!(at <= self.len()); self.split_off_checked(at)
.expect("at value must be <= self.len()`")
}
if at == self.len() { /// Splits the bytes into two at the given index.
return Bytes::new(); ///
} /// Does nothing if `at > self.len()`
pub fn split_off_checked(&mut self, at: usize) -> Option<Bytes> {
if at == 0 { if at <= self.len() {
mem::replace(self, Bytes::new()) if at == self.len() {
} else { Some(Bytes::new())
Bytes { } else if at == 0 {
inner: self.inner.split_off(at, true), Some(mem::replace(self, Bytes::new()))
} else {
Some(Bytes {
inner: self.inner.split_off(at, true),
})
} }
} else {
None
} }
} }
@ -686,18 +708,26 @@ impl Bytes {
/// ///
/// Panics if `at > len`. /// Panics if `at > len`.
pub fn split_to(&mut self, at: usize) -> Bytes { pub fn split_to(&mut self, at: usize) -> Bytes {
assert!(at <= self.len()); self.split_to_checked(at)
.expect("at value must be <= self.len()`")
}
if at == self.len() { /// Splits the bytes into two at the given index.
return mem::replace(self, Bytes::new()); ///
} /// Does nothing if `at > len`.
pub fn split_to_checked(&mut self, at: usize) -> Option<Bytes> {
if at == 0 { if at <= self.len() {
Bytes::new() if at == self.len() {
} else { Some(mem::replace(self, Bytes::new()))
Bytes { } else if at == 0 {
inner: self.inner.split_to(at, true), Some(Bytes::new())
} else {
Some(Bytes {
inner: self.inner.split_to(at, true),
})
} }
} else {
None
} }
} }
@ -1299,8 +1329,7 @@ impl BytesMut {
/// assert_eq!(other, b"hello world"[..]); /// assert_eq!(other, b"hello world"[..]);
/// ``` /// ```
pub fn split(&mut self) -> BytesMut { pub fn split(&mut self) -> BytesMut {
let len = self.len(); self.split_to(self.len())
self.split_to(len)
} }
/// Splits the buffer into two at the given index. /// Splits the buffer into two at the given index.
@ -1330,10 +1359,20 @@ impl BytesMut {
/// ///
/// Panics if `at > len`. /// Panics if `at > len`.
pub fn split_to(&mut self, at: usize) -> BytesMut { pub fn split_to(&mut self, at: usize) -> BytesMut {
assert!(at <= self.len()); self.split_to_checked(at)
.expect("at value must be <= self.len()`")
}
BytesMut { /// Splits the bytes into two at the given index.
inner: self.inner.split_to(at, false), ///
/// Does nothing if `at > len`.
pub fn split_to_checked(&mut self, at: usize) -> Option<BytesMut> {
if at <= self.len() {
Some(BytesMut {
inner: self.inner.split_to(at, false),
})
} else {
None
} }
} }
@ -2136,10 +2175,20 @@ impl BytesVec {
/// ///
/// Panics if `at > len`. /// Panics if `at > len`.
pub fn split_to(&mut self, at: usize) -> BytesMut { pub fn split_to(&mut self, at: usize) -> BytesMut {
assert!(at <= self.len()); self.split_to_checked(at)
.expect("at value must be <= self.len()`")
}
BytesMut { /// Splits the bytes into two at the given index.
inner: self.inner.split_to(at, false), ///
/// Does nothing if `at > len`.
pub fn split_to_checked(&mut self, at: usize) -> Option<BytesMut> {
if at <= self.len() {
Some(BytesMut {
inner: self.inner.split_to(at, false),
})
} else {
None
} }
} }

View file

@ -17,7 +17,7 @@ path = "src/lib.rs"
[dependencies] [dependencies]
ntex-codec = "0.6.2" ntex-codec = "0.6.2"
ntex-bytes = "0.1.21" ntex-bytes = "0.1.24"
ntex-util = "1.0.0" ntex-util = "1.0.0"
ntex-service = "2.0.0" ntex-service = "2.0.0"

View file

@ -55,7 +55,7 @@ ntex-router = "0.5.3"
ntex-service = "2.0.0" ntex-service = "2.0.0"
ntex-macros = "0.1.3" ntex-macros = "0.1.3"
ntex-util = "1.0.0" ntex-util = "1.0.0"
ntex-bytes = "0.1.23" ntex-bytes = "0.1.24"
ntex-h2 = "0.5.0" ntex-h2 = "0.5.0"
ntex-rt = "0.4.11" ntex-rt = "0.4.11"
ntex-io = "1.0.0" ntex-io = "1.0.0"