use std::{collections::HashMap, io::BufRead};
use crate::Error;
pub trait TokenResolver {
fn resolve(&self, token: u16) -> Option<&str>;
fn lookup(&self, _index: u32) -> Option<&str> {
None
}
fn is_empty(&self) -> bool {
false
}
}
impl<S, V> TokenResolver for HashMap<u16, V, S>
where
S: ::std::hash::BuildHasher,
V: AsRef<str>,
{
fn resolve(&self, token: u16) -> Option<&str> {
self.get(&token).map(|x| x.as_ref())
}
fn is_empty(&self) -> bool {
self.is_empty()
}
}
impl<T: TokenResolver> TokenResolver for &'_ T {
fn resolve(&self, token: u16) -> Option<&str> {
(**self).resolve(token)
}
fn lookup(&self, index: u32) -> Option<&str> {
(**self).lookup(index)
}
fn is_empty(&self) -> bool {
(**self).is_empty()
}
}
impl<T: TokenResolver + ?Sized> TokenResolver for Box<T> {
fn resolve(&self, token: u16) -> Option<&str> {
(**self).resolve(token)
}
fn lookup(&self, index: u32) -> Option<&str> {
(**self).lookup(index)
}
fn is_empty(&self) -> bool {
(**self).is_empty()
}
}
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum FailedResolveStrategy {
Error,
Stringify,
Ignore,
}
pub struct BasicTokenResolver {
lookup: HashMap<u16, String>,
}
impl BasicTokenResolver {
pub fn from_text_lines<T>(mut reader: T) -> Result<Self, Error>
where
T: BufRead,
{
let mut lookup = HashMap::new();
let mut line = String::new();
let mut pos = 0;
while reader.read_line(&mut line)? != 0 {
let (num, text) = line
.split_once(' ')
.ok_or_else(|| Error::invalid_syntax("expected to split line", pos))?;
let z = u16::from_str_radix(num.trim_start_matches("0x"), 16)
.map_err(|_| Error::invalid_syntax("invalid ironman token", pos))?;
pos += line.len();
lookup.insert(z, String::from(text.trim_ascii_end()));
line.clear();
}
Ok(Self { lookup })
}
}
impl TokenResolver for BasicTokenResolver {
fn resolve(&self, token: u16) -> Option<&str> {
self.lookup.get(&token).map(|x| x.as_str())
}
fn is_empty(&self) -> bool {
self.lookup.is_empty()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn can_create_resolve() {
let data = b"0xffff my_test_token\n0xeeee my_test_token2";
let resolver = BasicTokenResolver::from_text_lines(&data[..]).unwrap();
assert_eq!(resolver.resolve(0xffff), Some("my_test_token"));
assert_eq!(resolver.resolve(0xeeee), Some("my_test_token2"));
}
}