use memchr::memchr2;
use regex::{Regex, RegexBuilder};
use crate::style::Style;
use super::super::span::{Collector, Finder};
#[derive(Debug)]
pub(crate) struct PointerFinder {
regex: Regex,
number: Style,
letter: Style,
x: Style,
}
impl PointerFinder {
pub fn new(number: Style, letter: Style, x: Style) -> Self {
let pattern = r"(?i)\b0x(?:[0-9a-f]{16}|[0-9a-f]{8})\b";
let regex = RegexBuilder::new(pattern)
.unicode(false)
.build()
.expect("hardcoded pointer regex must compile");
Self {
regex,
number,
letter,
x,
}
}
}
impl Finder for PointerFinder {
fn find_spans(&self, input: &str, collector: &mut Collector) {
if memchr2(b'x', b'X', input.as_bytes()).is_none() {
return;
}
for m in self.regex.find_iter(input) {
let offset = m.start();
for (i, &b) in m.as_str().as_bytes().iter().enumerate() {
let style = match b {
b'0'..=b'9' => self.number,
b'x' | b'X' => self.x,
b'a'..=b'f' | b'A'..=b'F' => self.letter,
_ => continue,
};
collector.push(offset + i, offset + i + 1, style);
}
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::style::Color;
fn make_finder() -> PointerFinder {
PointerFinder::new(
Style::new().fg(Color::Blue),
Style::new().fg(Color::Magenta),
Style::new().fg(Color::Red),
)
}
fn span_count(input: &str) -> usize {
let mut collector = Collector::new(0);
make_finder().find_spans(input, &mut collector);
collector.into_spans().len()
}
fn matched_range(input: &str) -> Option<(usize, usize)> {
let mut collector = Collector::new(0);
make_finder().find_spans(input, &mut collector);
let spans = collector.into_spans();
if spans.is_empty() {
None
} else {
Some((spans.first().unwrap().start, spans.last().unwrap().end))
}
}
#[test]
fn pointer_32bit() {
let (start, end) = matched_range("0x8c2a0aeb").unwrap();
assert_eq!(start, 0);
assert_eq!(end, 10); }
#[test]
fn pointer_64bit() {
let (start, end) = matched_range("0xd7b3b2f446e2c21b").unwrap();
assert_eq!(start, 0);
assert_eq!(end, 18); }
#[test]
fn pointer_distinguishes_digits_and_letters() {
let finder = make_finder();
let input = "0xab12cd34";
let mut collector = Collector::new(0);
finder.find_spans(input, &mut collector);
let spans = collector.into_spans();
assert!(spans.len() >= 4);
}
#[test]
fn no_pointer_no_match() {
assert_eq!(span_count("No numbers here!"), 0);
}
}