pub mod printf { use super::strcursor::StrCursor as Cur; use syntax_pos::InnerSpan; /// Represents a single `printf`-style substitution. #[derive(Clone, PartialEq, Debug)] pub enum Substitution<'a> { /// A formatted output substitution with its internal byte offset. Format(Format<'a>), /// A literal `%%` escape. Escape, } impl<'a> Substitution<'a> { pub fn as_str(&self) -> &str { match *self { Substitution::Format(ref fmt) => fmt.span, Substitution::Escape => "%%", } } pub fn position(&self) -> Option { match *self { Substitution::Format(ref fmt) => Some(fmt.position), _ => None, } } pub fn set_position(&mut self, start: usize, end: usize) { match self { Substitution::Format(ref mut fmt) => { fmt.position = InnerSpan::new(start, end); } _ => {} } } /// Translate this substitution into an equivalent Rust formatting directive. /// /// This ignores cases where the substitution does not have an exact equivalent, or where /// the substitution would be unnecessary. pub fn translate(&self) -> Option { match *self { Substitution::Format(ref fmt) => fmt.translate(), Substitution::Escape => None, } } } #[derive(Clone, PartialEq, Debug)] /// A single `printf`-style formatting directive. pub struct Format<'a> { /// The entire original formatting directive. pub span: &'a str, /// The (1-based) parameter to be converted. pub parameter: Option, /// Formatting flags. pub flags: &'a str, /// Minimum width of the output. pub width: Option, /// Precision of the conversion. pub precision: Option, /// Length modifier for the conversion. pub length: Option<&'a str>, /// Type of parameter being converted. pub type_: &'a str, /// Byte offset for the start and end of this formatting directive. pub position: InnerSpan, } impl Format<'_> { /// Translate this directive into an equivalent Rust formatting directive. /// /// Returns `None` in cases where the `printf` directive does not have an exact Rust /// equivalent, rather than guessing. pub fn translate(&self) -> Option { use std::fmt::Write; let (c_alt, c_zero, c_left, c_plus) = { let mut c_alt = false; let mut c_zero = false; let mut c_left = false; let mut c_plus = false; for c in self.flags.chars() { match c { '#' => c_alt = true, '0' => c_zero = true, '-' => c_left = true, '+' => c_plus = true, _ => return None } } (c_alt, c_zero, c_left, c_plus) }; // Has a special form in Rust for numbers. let fill = if c_zero { Some("0") } else { None }; let align = if c_left { Some("<") } else { None }; // Rust doesn't have an equivalent to the `' '` flag. let sign = if c_plus { Some("+") } else { None }; // Not *quite* the same, depending on the type... let alt = c_alt; let width = match self.width { Some(Num::Next) => { // NOTE: Rust doesn't support this. return None; } w @ Some(Num::Arg(_)) => w, w @ Some(Num::Num(_)) => w, None => None, }; let precision = self.precision; // NOTE: although length *can* have an effect, we can't duplicate the effect in Rust, so // we just ignore it. let (type_, use_zero_fill, is_int) = match self.type_ { "d" | "i" | "u" => (None, true, true), "f" | "F" => (None, false, false), "s" | "c" => (None, false, false), "e" | "E" => (Some(self.type_), true, false), "x" | "X" | "o" => (Some(self.type_), true, true), "p" => (Some(self.type_), false, true), "g" => (Some("e"), true, false), "G" => (Some("E"), true, false), _ => return None, }; let (fill, width, precision) = match (is_int, width, precision) { (true, Some(_), Some(_)) => { // Rust can't duplicate this insanity. return None; }, (true, None, Some(p)) => (Some("0"), Some(p), None), (true, w, None) => (fill, w, None), (false, w, p) => (fill, w, p), }; let align = match (self.type_, width.is_some(), align.is_some()) { ("s", true, false) => Some(">"), _ => align, }; let (fill, zero_fill) = match (fill, use_zero_fill) { (Some("0"), true) => (None, true), (fill, _) => (fill, false), }; let alt = match type_ { Some("x") | Some("X") => alt, _ => false, }; let has_options = fill.is_some() || align.is_some() || sign.is_some() || alt || zero_fill || width.is_some() || precision.is_some() || type_.is_some() ; // Initialise with a rough guess. let cap = self.span.len() + if has_options { 2 } else { 0 }; let mut s = String::with_capacity(cap); s.push_str("{"); if let Some(arg) = self.parameter { write!(s, "{}", arg.checked_sub(1)?).ok()?; } if has_options { s.push_str(":"); let align = if let Some(fill) = fill { s.push_str(fill); align.or(Some(">")) } else { align }; if let Some(align) = align { s.push_str(align); } if let Some(sign) = sign { s.push_str(sign); } if alt { s.push_str("#"); } if zero_fill { s.push_str("0"); } if let Some(width) = width { width.translate(&mut s).ok()?; } if let Some(precision) = precision { s.push_str("."); precision.translate(&mut s).ok()?; } if let Some(type_) = type_ { s.push_str(type_); } } s.push_str("}"); Some(s) } } /// A general number used in a `printf` formatting directive. #[derive(Copy, Clone, PartialEq, Debug)] pub enum Num { // The range of these values is technically bounded by `NL_ARGMAX`... but, at least for GNU // libc, it apparently has no real fixed limit. A `u16` is used here on the basis that it // is *vanishingly* unlikely that *anyone* is going to try formatting something wider, or // with more precision, than 32 thousand positions which is so wide it couldn't possibly fit // on a screen. /// A specific, fixed value. Num(u16), /// The value is derived from a positional argument. Arg(u16), /// The value is derived from the "next" unconverted argument. Next, } impl Num { fn from_str(s: &str, arg: Option<&str>) -> Self { if let Some(arg) = arg { Num::Arg(arg.parse().unwrap_or_else(|_| panic!("invalid format arg `{:?}`", arg))) } else if s == "*" { Num::Next } else { Num::Num(s.parse().unwrap_or_else(|_| panic!("invalid format num `{:?}`", s))) } } fn translate(&self, s: &mut String) -> std::fmt::Result { use std::fmt::Write; match *self { Num::Num(n) => write!(s, "{}", n), Num::Arg(n) => { let n = n.checked_sub(1).ok_or(std::fmt::Error)?; write!(s, "{}$", n) }, Num::Next => write!(s, "*"), } } } /// Returns an iterator over all substitutions in a given string. pub fn iter_subs(s: &str, start_pos: usize) -> Substitutions<'_> { Substitutions { s, pos: start_pos, } } /// Iterator over substitutions in a string. pub struct Substitutions<'a> { s: &'a str, pos: usize, } impl<'a> Iterator for Substitutions<'a> { type Item = Substitution<'a>; fn next(&mut self) -> Option { let (mut sub, tail) = parse_next_substitution(self.s)?; self.s = tail; match sub { Substitution::Format(_) => if let Some(inner_span) = sub.position() { sub.set_position(inner_span.start + self.pos, inner_span.end + self.pos); self.pos += inner_span.end; } Substitution::Escape => self.pos += 2, } Some(sub) } fn size_hint(&self) -> (usize, Option) { // Substitutions are at least 2 characters long. (0, Some(self.s.len() / 2)) } } enum State { Start, Flags, Width, WidthArg, Prec, PrecInner, Length, Type, } /// Parse the next substitution from the input string. pub fn parse_next_substitution(s: &str) -> Option<(Substitution<'_>, &str)> { use self::State::*; let at = { let start = s.find('%')?; match s[start+1..].chars().next()? { '%' => return Some((Substitution::Escape, &s[start+2..])), _ => {/* fall-through */}, } Cur::new_at(&s[..], start) }; // This is meant to be a translation of the following regex: // // ```regex // (?x) // ^ % // (?: (?P \d+) \$ )? // (?P [-+ 0\#']* ) // (?P \d+ | \* (?: (?P \d+) \$ )? )? // (?: \. (?P \d+ | \* (?: (?P \d+) \$ )? ) )? // (?P // # Standard // hh | h | ll | l | L | z | j | t // // # Other // | I32 | I64 | I | q // )? // (?P . ) // ``` // Used to establish the full span at the end. let start = at; // The current position within the string. let mut at = at.at_next_cp()?; // `c` is the next codepoint, `next` is a cursor after it. let (mut c, mut next) = at.next_cp()?; // Update `at`, `c`, and `next`, exiting if we're out of input. macro_rules! move_to { ($cur:expr) => { { at = $cur; let (c_, next_) = at.next_cp()?; c = c_; next = next_; } }; } // Constructs a result when parsing fails. // // Note: `move` used to capture copies of the cursors as they are *now*. let fallback = move || { return Some(( Substitution::Format(Format { span: start.slice_between(next).unwrap(), parameter: None, flags: "", width: None, precision: None, length: None, type_: at.slice_between(next).unwrap(), position: InnerSpan::new(start.at, next.at), }), next.slice_after() )); }; // Next parsing state. let mut state = Start; // Sadly, Rust isn't *quite* smart enough to know these *must* be initialised by the end. let mut parameter: Option = None; let mut flags: &str = ""; let mut width: Option = None; let mut precision: Option = None; let mut length: Option<&str> = None; let mut type_: &str = ""; let end: Cur<'_>; if let Start = state { match c { '1'..='9' => { let end = at_next_cp_while(next, is_digit); match end.next_cp() { // Yes, this *is* the parameter. Some(('$', end2)) => { state = Flags; parameter = Some(at.slice_between(end).unwrap().parse().unwrap()); move_to!(end2); }, // Wait, no, actually, it's the width. Some(_) => { state = Prec; parameter = None; flags = ""; width = Some(Num::from_str(at.slice_between(end).unwrap(), None)); move_to!(end); }, // It's invalid, is what it is. None => return fallback(), } }, _ => { state = Flags; parameter = None; move_to!(at); } } } if let Flags = state { let end = at_next_cp_while(at, is_flag); state = Width; flags = at.slice_between(end).unwrap(); move_to!(end); } if let Width = state { match c { '*' => { state = WidthArg; move_to!(next); }, '1' ..= '9' => { let end = at_next_cp_while(next, is_digit); state = Prec; width = Some(Num::from_str(at.slice_between(end).unwrap(), None)); move_to!(end); }, _ => { state = Prec; width = None; move_to!(at); } } } if let WidthArg = state { let end = at_next_cp_while(at, is_digit); match end.next_cp() { Some(('$', end2)) => { state = Prec; width = Some(Num::from_str("", Some(at.slice_between(end).unwrap()))); move_to!(end2); }, _ => { state = Prec; width = Some(Num::Next); move_to!(end); } } } if let Prec = state { match c { '.' => { state = PrecInner; move_to!(next); }, _ => { state = Length; precision = None; move_to!(at); } } } if let PrecInner = state { match c { '*' => { let end = at_next_cp_while(next, is_digit); match end.next_cp() { Some(('$', end2)) => { state = Length; precision = Some(Num::from_str("*", next.slice_between(end))); move_to!(end2); }, _ => { state = Length; precision = Some(Num::Next); move_to!(end); } } }, '0' ..= '9' => { let end = at_next_cp_while(next, is_digit); state = Length; precision = Some(Num::from_str(at.slice_between(end).unwrap(), None)); move_to!(end); }, _ => return fallback(), } } if let Length = state { let c1_next1 = next.next_cp(); match (c, c1_next1) { ('h', Some(('h', next1))) | ('l', Some(('l', next1))) => { state = Type; length = Some(at.slice_between(next1).unwrap()); move_to!(next1); }, ('h', _) | ('l', _) | ('L', _) | ('z', _) | ('j', _) | ('t', _) | ('q', _) => { state = Type; length = Some(at.slice_between(next).unwrap()); move_to!(next); }, ('I', _) => { let end = next.at_next_cp() .and_then(|end| end.at_next_cp()) .map(|end| (next.slice_between(end).unwrap(), end)); let end = match end { Some(("32", end)) => end, Some(("64", end)) => end, _ => next }; state = Type; length = Some(at.slice_between(end).unwrap()); move_to!(end); }, _ => { state = Type; length = None; move_to!(at); } } } if let Type = state { drop(c); type_ = at.slice_between(next).unwrap(); // Don't use `move_to!` here, as we *can* be at the end of the input. at = next; } drop(c); drop(next); end = at; let position = InnerSpan::new(start.at, end.at); let f = Format { span: start.slice_between(end).unwrap(), parameter, flags, width, precision, length, type_, position, }; Some((Substitution::Format(f), end.slice_after())) } fn at_next_cp_while(mut cur: Cur<'_>, mut pred: F) -> Cur<'_> where F: FnMut(char) -> bool { loop { match cur.next_cp() { Some((c, next)) => if pred(c) { cur = next; } else { return cur; }, None => return cur, } } } fn is_digit(c: char) -> bool { match c { '0' ..= '9' => true, _ => false } } fn is_flag(c: char) -> bool { match c { '0' | '-' | '+' | ' ' | '#' | '\'' => true, _ => false } } #[cfg(test)] mod tests { use super::{ Format as F, Num as N, Substitution as S, iter_subs, parse_next_substitution as pns, }; macro_rules! assert_eq_pnsat { ($lhs:expr, $rhs:expr) => { assert_eq!( pns($lhs).and_then(|(s, _)| s.translate()), $rhs.map(>::from) ) }; } #[test] fn test_escape() { assert_eq!(pns("has no escapes"), None); assert_eq!(pns("has no escapes, either %"), None); assert_eq!(pns("*so* has a %% escape"), Some((S::Escape," escape"))); assert_eq!(pns("%% leading escape"), Some((S::Escape, " leading escape"))); assert_eq!(pns("trailing escape %%"), Some((S::Escape, ""))); } #[test] fn test_parse() { macro_rules! assert_pns_eq_sub { ($in_:expr, { $param:expr, $flags:expr, $width:expr, $prec:expr, $len:expr, $type_:expr, $pos:expr, }) => { assert_eq!( pns(concat!($in_, "!")), Some(( S::Format(F { span: $in_, parameter: $param, flags: $flags, width: $width, precision: $prec, length: $len, type_: $type_, position: syntax_pos::InnerSpan::new($pos.0, $pos.1), }), "!" )) ) }; } assert_pns_eq_sub!("%!", { None, "", None, None, None, "!", (0, 2), }); assert_pns_eq_sub!("%c", { None, "", None, None, None, "c", (0, 2), }); assert_pns_eq_sub!("%s", { None, "", None, None, None, "s", (0, 2), }); assert_pns_eq_sub!("%06d", { None, "0", Some(N::Num(6)), None, None, "d", (0, 4), }); assert_pns_eq_sub!("%4.2f", { None, "", Some(N::Num(4)), Some(N::Num(2)), None, "f", (0, 5), }); assert_pns_eq_sub!("%#x", { None, "#", None, None, None, "x", (0, 3), }); assert_pns_eq_sub!("%-10s", { None, "-", Some(N::Num(10)), None, None, "s", (0, 5), }); assert_pns_eq_sub!("%*s", { None, "", Some(N::Next), None, None, "s", (0, 3), }); assert_pns_eq_sub!("%-10.*s", { None, "-", Some(N::Num(10)), Some(N::Next), None, "s", (0, 7), }); assert_pns_eq_sub!("%-*.*s", { None, "-", Some(N::Next), Some(N::Next), None, "s", (0, 6), }); assert_pns_eq_sub!("%.6i", { None, "", None, Some(N::Num(6)), None, "i", (0, 4), }); assert_pns_eq_sub!("%+i", { None, "+", None, None, None, "i", (0, 3), }); assert_pns_eq_sub!("%08X", { None, "0", Some(N::Num(8)), None, None, "X", (0, 4), }); assert_pns_eq_sub!("%lu", { None, "", None, None, Some("l"), "u", (0, 3), }); assert_pns_eq_sub!("%Iu", { None, "", None, None, Some("I"), "u", (0, 3), }); assert_pns_eq_sub!("%I32u", { None, "", None, None, Some("I32"), "u", (0, 5), }); assert_pns_eq_sub!("%I64u", { None, "", None, None, Some("I64"), "u", (0, 5), }); assert_pns_eq_sub!("%'d", { None, "'", None, None, None, "d", (0, 3), }); assert_pns_eq_sub!("%10s", { None, "", Some(N::Num(10)), None, None, "s", (0, 4), }); assert_pns_eq_sub!("%-10.10s", { None, "-", Some(N::Num(10)), Some(N::Num(10)), None, "s", (0, 8), }); assert_pns_eq_sub!("%1$d", { Some(1), "", None, None, None, "d", (0, 4), }); assert_pns_eq_sub!("%2$.*3$d", { Some(2), "", None, Some(N::Arg(3)), None, "d", (0, 8), }); assert_pns_eq_sub!("%1$*2$.*3$d", { Some(1), "", Some(N::Arg(2)), Some(N::Arg(3)), None, "d", (0, 11), }); assert_pns_eq_sub!("%-8ld", { None, "-", Some(N::Num(8)), None, Some("l"), "d", (0, 5), }); } #[test] fn test_iter() { let s = "The %d'th word %% is: `%.*s` %!\n"; let subs: Vec<_> = iter_subs(s, 0).map(|sub| sub.translate()).collect(); assert_eq!( subs.iter().map(|ms| ms.as_ref().map(|s| &s[..])).collect::>(), vec![Some("{}"), None, Some("{:.*}"), None] ); } /// Checks that the translations are what we expect. #[test] fn test_translation() { assert_eq_pnsat!("%c", Some("{}")); assert_eq_pnsat!("%d", Some("{}")); assert_eq_pnsat!("%u", Some("{}")); assert_eq_pnsat!("%x", Some("{:x}")); assert_eq_pnsat!("%X", Some("{:X}")); assert_eq_pnsat!("%e", Some("{:e}")); assert_eq_pnsat!("%E", Some("{:E}")); assert_eq_pnsat!("%f", Some("{}")); assert_eq_pnsat!("%g", Some("{:e}")); assert_eq_pnsat!("%G", Some("{:E}")); assert_eq_pnsat!("%s", Some("{}")); assert_eq_pnsat!("%p", Some("{:p}")); assert_eq_pnsat!("%06d", Some("{:06}")); assert_eq_pnsat!("%4.2f", Some("{:4.2}")); assert_eq_pnsat!("%#x", Some("{:#x}")); assert_eq_pnsat!("%-10s", Some("{:<10}")); assert_eq_pnsat!("%*s", None); assert_eq_pnsat!("%-10.*s", Some("{:<10.*}")); assert_eq_pnsat!("%-*.*s", None); assert_eq_pnsat!("%.6i", Some("{:06}")); assert_eq_pnsat!("%+i", Some("{:+}")); assert_eq_pnsat!("%08X", Some("{:08X}")); assert_eq_pnsat!("%lu", Some("{}")); assert_eq_pnsat!("%Iu", Some("{}")); assert_eq_pnsat!("%I32u", Some("{}")); assert_eq_pnsat!("%I64u", Some("{}")); assert_eq_pnsat!("%'d", None); assert_eq_pnsat!("%10s", Some("{:>10}")); assert_eq_pnsat!("%-10.10s", Some("{:<10.10}")); assert_eq_pnsat!("%1$d", Some("{0}")); assert_eq_pnsat!("%2$.*3$d", Some("{1:02$}")); assert_eq_pnsat!("%1$*2$.*3$s", Some("{0:>1$.2$}")); assert_eq_pnsat!("%-8ld", Some("{:<8}")); } } } pub mod shell { use super::strcursor::StrCursor as Cur; use syntax_pos::InnerSpan; #[derive(Clone, PartialEq, Debug)] pub enum Substitution<'a> { Ordinal(u8, (usize, usize)), Name(&'a str, (usize, usize)), Escape((usize, usize)), } impl Substitution<'_> { pub fn as_str(&self) -> String { match self { Substitution::Ordinal(n, _) => format!("${}", n), Substitution::Name(n, _) => format!("${}", n), Substitution::Escape(_) => "$$".into(), } } pub fn position(&self) -> Option { match self { Substitution::Ordinal(_, pos) | Substitution::Name(_, pos) | Substitution::Escape(pos) => Some(InnerSpan::new(pos.0, pos.1)), } } pub fn set_position(&mut self, start: usize, end: usize) { match self { Substitution::Ordinal(_, ref mut pos) | Substitution::Name(_, ref mut pos) | Substitution::Escape(ref mut pos) => *pos = (start, end), } } pub fn translate(&self) -> Option { match *self { Substitution::Ordinal(n, _) => Some(format!("{{{}}}", n)), Substitution::Name(n, _) => Some(format!("{{{}}}", n)), Substitution::Escape(_) => None, } } } /// Returns an iterator over all substitutions in a given string. pub fn iter_subs(s: &str, start_pos: usize) -> Substitutions<'_> { Substitutions { s, pos: start_pos, } } /// Iterator over substitutions in a string. pub struct Substitutions<'a> { s: &'a str, pos: usize, } impl<'a> Iterator for Substitutions<'a> { type Item = Substitution<'a>; fn next(&mut self) -> Option { match parse_next_substitution(self.s) { Some((mut sub, tail)) => { self.s = tail; if let Some(InnerSpan { start, end }) = sub.position() { sub.set_position(start + self.pos, end + self.pos); self.pos += end; } Some(sub) }, None => None, } } fn size_hint(&self) -> (usize, Option) { (0, Some(self.s.len())) } } /// Parse the next substitution from the input string. pub fn parse_next_substitution(s: &str) -> Option<(Substitution<'_>, &str)> { let at = { let start = s.find('$')?; match s[start+1..].chars().next()? { '$' => return Some((Substitution::Escape((start, start+2)), &s[start+2..])), c @ '0' ..= '9' => { let n = (c as u8) - b'0'; return Some((Substitution::Ordinal(n, (start, start+2)), &s[start+2..])); }, _ => {/* fall-through */}, } Cur::new_at(&s[..], start) }; let at = at.at_next_cp()?; let (c, inner) = at.next_cp()?; if !is_ident_head(c) { None } else { let end = at_next_cp_while(inner, is_ident_tail); let slice = at.slice_between(end).unwrap(); let start = at.at - 1; let end_pos = at.at + slice.len(); Some((Substitution::Name(slice, (start, end_pos)), end.slice_after())) } } fn at_next_cp_while(mut cur: Cur<'_>, mut pred: F) -> Cur<'_> where F: FnMut(char) -> bool { loop { match cur.next_cp() { Some((c, next)) => if pred(c) { cur = next; } else { return cur; }, None => return cur, } } } fn is_ident_head(c: char) -> bool { match c { 'a' ..= 'z' | 'A' ..= 'Z' | '_' => true, _ => false } } fn is_ident_tail(c: char) -> bool { match c { '0' ..= '9' => true, c => is_ident_head(c) } } #[cfg(test)] mod tests { use super::{ Substitution as S, parse_next_substitution as pns, }; macro_rules! assert_eq_pnsat { ($lhs:expr, $rhs:expr) => { assert_eq!( pns($lhs).and_then(|(f, _)| f.translate()), $rhs.map(>::from) ) }; } #[test] fn test_escape() { assert_eq!(pns("has no escapes"), None); assert_eq!(pns("has no escapes, either $"), None); assert_eq!(pns("*so* has a $$ escape"), Some((S::Escape((11, 13)), " escape"))); assert_eq!(pns("$$ leading escape"), Some((S::Escape((0, 2)), " leading escape"))); assert_eq!(pns("trailing escape $$"), Some((S::Escape((16, 18)), ""))); } #[test] fn test_parse() { macro_rules! assert_pns_eq_sub { ($in_:expr, $kind:ident($arg:expr, $pos:expr)) => { assert_eq!(pns(concat!($in_, "!")), Some((S::$kind($arg.into(), $pos), "!"))) }; } assert_pns_eq_sub!("$0", Ordinal(0, (0, 2))); assert_pns_eq_sub!("$1", Ordinal(1, (0, 2))); assert_pns_eq_sub!("$9", Ordinal(9, (0, 2))); assert_pns_eq_sub!("$N", Name("N", (0, 2))); assert_pns_eq_sub!("$NAME", Name("NAME", (0, 5))); } #[test] fn test_iter() { use super::iter_subs; let s = "The $0'th word $$ is: `$WORD` $!\n"; let subs: Vec<_> = iter_subs(s, 0).map(|sub| sub.translate()).collect(); assert_eq!( subs.iter().map(|ms| ms.as_ref().map(|s| &s[..])).collect::>(), vec![Some("{0}"), None, Some("{WORD}")] ); } #[test] fn test_translation() { assert_eq_pnsat!("$0", Some("{0}")); assert_eq_pnsat!("$9", Some("{9}")); assert_eq_pnsat!("$1", Some("{1}")); assert_eq_pnsat!("$10", Some("{1}")); assert_eq_pnsat!("$stuff", Some("{stuff}")); assert_eq_pnsat!("$NAME", Some("{NAME}")); assert_eq_pnsat!("$PREFIX/bin", Some("{PREFIX}")); } } } mod strcursor { pub struct StrCursor<'a> { s: &'a str, pub at: usize, } impl<'a> StrCursor<'a> { pub fn new_at(s: &'a str, at: usize) -> StrCursor<'a> { StrCursor { s, at, } } pub fn at_next_cp(mut self) -> Option> { match self.try_seek_right_cp() { true => Some(self), false => None } } pub fn next_cp(mut self) -> Option<(char, StrCursor<'a>)> { let cp = self.cp_after()?; self.seek_right(cp.len_utf8()); Some((cp, self)) } fn slice_before(&self) -> &'a str { &self.s[0..self.at] } pub fn slice_after(&self) -> &'a str { &self.s[self.at..] } pub fn slice_between(&self, until: StrCursor<'a>) -> Option<&'a str> { if !str_eq_literal(self.s, until.s) { None } else { use std::cmp::{max, min}; let beg = min(self.at, until.at); let end = max(self.at, until.at); Some(&self.s[beg..end]) } } fn cp_after(&self) -> Option { self.slice_after().chars().next() } fn try_seek_right_cp(&mut self) -> bool { match self.slice_after().chars().next() { Some(c) => { self.at += c.len_utf8(); true }, None => false, } } fn seek_right(&mut self, bytes: usize) { self.at += bytes; } } impl Copy for StrCursor<'_> {} impl<'a> Clone for StrCursor<'a> { fn clone(&self) -> StrCursor<'a> { *self } } impl std::fmt::Debug for StrCursor<'_> { fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(fmt, "StrCursor({:?} | {:?})", self.slice_before(), self.slice_after()) } } fn str_eq_literal(a: &str, b: &str) -> bool { a.as_bytes().as_ptr() == b.as_bytes().as_ptr() && a.len() == b.len() } }