mirror of
https://github.com/clap-rs/clap
synced 2024-12-14 14:52:33 +00:00
Optimize ArgStr
This commit is contained in:
parent
216aab2479
commit
5f70e38c2a
1 changed files with 101 additions and 33 deletions
|
@ -7,6 +7,7 @@ use std::{
|
||||||
|
|
||||||
use os_str_bytes::{raw, OsStrBytes};
|
use os_str_bytes::{raw, OsStrBytes};
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq)]
|
||||||
pub(crate) struct ArgStr<'a>(Cow<'a, [u8]>);
|
pub(crate) struct ArgStr<'a>(Cow<'a, [u8]>);
|
||||||
|
|
||||||
impl<'a> ArgStr<'a> {
|
impl<'a> ArgStr<'a> {
|
||||||
|
@ -27,58 +28,47 @@ impl<'a> ArgStr<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn contains_byte(&self, byte: u8) -> bool {
|
pub(crate) fn contains_byte(&self, byte: u8) -> bool {
|
||||||
assert!(byte.is_ascii());
|
debug_assert!(byte.is_ascii());
|
||||||
|
|
||||||
self.0.contains(&byte)
|
self.0.contains(&byte)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn contains_char(&self, ch: char) -> bool {
|
pub(crate) fn contains_char(&self, ch: char) -> bool {
|
||||||
let mut bytes = [0; 4];
|
self.to_string_lossy().contains(|x| x == ch)
|
||||||
let bytes = ch.encode_utf8(&mut bytes).as_bytes();
|
|
||||||
for i in 0..self.0.len().saturating_sub(bytes.len() - 1) {
|
|
||||||
if self.0[i..].starts_with(bytes) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn split_at_byte(&self, byte: u8) -> (ArgStr, ArgStr) {
|
pub(crate) fn split_at_byte(&self, byte: u8) -> (ArgStr, ArgStr) {
|
||||||
assert!(byte.is_ascii());
|
debug_assert!(byte.is_ascii());
|
||||||
|
|
||||||
for (i, b) in self.0.iter().enumerate() {
|
if let Some(i) = self.0.iter().position(|&x| x == byte) {
|
||||||
if b == &byte {
|
self.split_at_unchecked(i)
|
||||||
return self.split_at_unchecked(i);
|
} else {
|
||||||
}
|
(self.to_borrowed(), Self(Cow::Borrowed(&[])))
|
||||||
}
|
}
|
||||||
(self.to_borrowed(), Self(Cow::Borrowed(&[])))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn trim_start_matches(&'a self, byte: u8) -> ArgStr {
|
pub(crate) fn trim_start_matches(&'a self, byte: u8) -> ArgStr {
|
||||||
assert!(byte.is_ascii());
|
debug_assert!(byte.is_ascii());
|
||||||
|
|
||||||
let mut found = false;
|
if let Some(i) = self.0.iter().position(|x| x != &byte) {
|
||||||
for (i, b) in self.0.iter().enumerate() {
|
Self(Cow::Borrowed(&self.0[i..]))
|
||||||
if b != &byte {
|
} else {
|
||||||
return Self(Cow::Borrowed(&self.0[i..]));
|
Self(Cow::Borrowed(&[]))
|
||||||
} else {
|
|
||||||
found = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if found {
|
|
||||||
return Self(Cow::Borrowed(&[]));
|
|
||||||
}
|
|
||||||
self.to_borrowed()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Like `trim_start_matches`, but trims no more than `n` matches
|
// Like `trim_start_matches`, but trims no more than `n` matches
|
||||||
#[inline]
|
pub(crate) fn trim_start_n_matches(&'a self, n: usize, byte: u8) -> ArgStr {
|
||||||
pub(crate) fn trim_start_n_matches(&self, n: usize, ch: u8) -> ArgStr {
|
debug_assert!(byte.is_ascii());
|
||||||
assert!(ch.is_ascii());
|
|
||||||
|
|
||||||
let i = self.0.iter().take(n).take_while(|c| **c == ch).count();
|
if let Some(i) = self.0.iter().take(n).position(|x| x != &byte) {
|
||||||
|
Self(Cow::Borrowed(&self.0[i..]))
|
||||||
self.split_at_unchecked(i).1
|
} else {
|
||||||
|
match self.0.get(n..) {
|
||||||
|
Some(x) => Self(Cow::Borrowed(x)),
|
||||||
|
None => Self(Cow::Borrowed(&[])),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn split_at_unchecked(&'a self, i: usize) -> (ArgStr, ArgStr) {
|
pub(crate) fn split_at_unchecked(&'a self, i: usize) -> (ArgStr, ArgStr) {
|
||||||
|
@ -188,3 +178,81 @@ impl<'a> Iterator for ArgSplit<'a> {
|
||||||
Some(ArgStr(Cow::Borrowed(&self.val[start..])))
|
Some(ArgStr(Cow::Borrowed(&self.val[start..])))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[rustfmt::skip]
|
||||||
|
fn test_trim_start_matches() {
|
||||||
|
let raw = OsString::from("hello? world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_matches(b'-');
|
||||||
|
assert_eq!(trimmed, a);
|
||||||
|
|
||||||
|
let raw = OsString::from("------------hello? world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_matches(b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hello? world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("------------hel-lo? -world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_matches(b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hel-lo? -world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("hel-lo? -world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_matches(b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hel-lo? -world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_matches(b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("")));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[rustfmt::skip]
|
||||||
|
fn test_trim_start_n_matches() {
|
||||||
|
let raw = OsString::from("hello? world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(2, b'-');
|
||||||
|
assert_eq!(trimmed, a);
|
||||||
|
|
||||||
|
let raw = OsString::from("------------hello? world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(2, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("----------hello? world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("------------hello? world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(1000, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hello? world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("------------hel-lo? -world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(2, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("----------hel-lo? -world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("-hel-lo? -world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(5, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hel-lo? -world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("hel-lo? -world");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(10, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("hel-lo? -world")));
|
||||||
|
|
||||||
|
let raw = OsString::from("");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(10, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("")));
|
||||||
|
|
||||||
|
let raw = OsString::from("");
|
||||||
|
let a = ArgStr::new(&raw);
|
||||||
|
let trimmed = a.trim_start_n_matches(0, b'-');
|
||||||
|
assert_eq!(trimmed, ArgStr::new(&OsString::from("")));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue