use regex::Regex;
use crate::{
BarcodeFormat, RXingResult, client::result::VINParsedRXingResult, common::Result,
exceptions::Exceptions,
};
use super::ParsedClientResult;
use once_cell::sync::Lazy;
static IOQ_MATCHER: Lazy<Regex> = Lazy::new(|| Regex::new(IOQ).unwrap());
static AZ09_MATCHER: Lazy<Regex> = Lazy::new(|| Regex::new(AZ09).unwrap());
pub fn parse(result: &RXingResult) -> Option<ParsedClientResult> {
if result.getBarcodeFormat() != &BarcodeFormat::CODE_39 {
return None;
}
let raw_text_res = result.getText().trim();
let raw_text = IOQ_MATCHER.replace_all(raw_text_res, "").to_string();
AZ09_MATCHER.find(&raw_text)?;
let check_cs = check_checksum(&raw_text).unwrap_or(false);
if !check_cs {
return None;
}
let wmi = &raw_text[..3];
let country_code = country_code(wmi).unwrap_or("");
let model_year = model_year(raw_text.chars().nth(9).unwrap_or('_')).ok();
Some(ParsedClientResult::VINResult(VINParsedRXingResult::new(
raw_text.to_owned(),
wmi.to_owned(),
raw_text[3..9].to_owned(),
raw_text[9..17].to_owned(),
country_code.to_owned(),
raw_text[3..8].to_owned(),
model_year?,
raw_text.chars().nth(10)?,
raw_text[11..].to_owned(),
)))
}
const IOQ: &str = "[IOQ]";
const AZ09: &str = "[A-Z0-9]{17}";
fn check_checksum(vin: &str) -> Result<bool> {
let mut sum = 0;
for i in 0..vin.len() {
sum += vin_position_weight(i + 1)? as u32
* vin_char_value(vin.chars().nth(i).ok_or(Exceptions::ILLEGAL_ARGUMENT)?)?;
}
let check_to_char = vin.chars().nth(8).ok_or(Exceptions::ILLEGAL_ARGUMENT)?;
let expected_check_char = check_char((sum % 11) as u8)?;
Ok(check_to_char == expected_check_char)
}
fn vin_char_value(c: char) -> Result<u32> {
match c {
'A'..='I' => Ok((c as u8 as u32 - b'A' as u32) + 1),
'J'..='R' => Ok((c as u8 as u32 - b'J' as u32) + 1),
'S'..='Z' => Ok((c as u8 as u32 - b'S' as u32) + 2),
'0'..='9' => Ok(c as u8 as u32 - b'0' as u32),
_ => Err(Exceptions::illegal_argument_with("vin char out of range")),
}
}
fn vin_position_weight(position: usize) -> Result<usize> {
match position {
1..=7 => Ok(9 - position),
8 => Ok(10),
9 => Ok(0),
10..=17 => Ok(19 - position),
_ => Err(Exceptions::illegal_argument_with(
"vin position weight out of bounds",
)),
}
}
fn check_char(remainder: u8) -> Result<char> {
match remainder {
0..=9 => Ok((b'0' + remainder) as char),
10 => Ok('X'),
_ => Err(Exceptions::illegal_argument_with("remainder too high")),
}
}
fn model_year(c: char) -> Result<u32> {
match c {
'E'..='H' => Ok((c as u8 as u32 - b'E' as u32) + 1984),
'J'..='N' => Ok((c as u8 as u32 - b'J' as u32) + 1988),
'P' => Ok(1993),
'R'..='T' => Ok((c as u8 as u32 - b'R' as u32) + 1994),
'V'..='Y' => Ok((c as u8 as u32 - b'V' as u32) + 1997),
'1'..='9' => Ok((c as u8 as u32 - b'1' as u32) + 2001),
'A'..='D' => Ok((c as u8 as u32 - b'A' as u32) + 2010),
_ => Err(Exceptions::illegal_argument_with(
"model year argument out of range",
)),
}
}
fn country_code(wmi: &str) -> Option<&'static str> {
let c1 = wmi.chars().next()?;
let c2 = wmi.chars().nth(1)?;
match c1 {
'1' | '4' | '5' => Some("US"),
'2' => Some("CA"),
'3' if ('A'..='W').contains(&c2) => Some("MX"),
'9' if (('A'..='E').contains(&c2) || ('3'..='9').contains(&c2)) => Some("BR"),
'J' if ('A'..='T').contains(&c2) => Some("JP"),
'K' if ('L'..='R').contains(&c2) => Some("KO"),
'L' => Some("CN"),
'M' if ('A'..='E').contains(&c2) => Some("IN"),
'S' if ('A'..='M').contains(&c2) => Some("UK"),
'S' if ('N'..='T').contains(&c2) => Some("DE"),
'V' if ('F'..='R').contains(&c2) => Some("FR"),
'V' if ('S'..='W').contains(&c2) => Some("ES"),
'W' => Some("DE"),
'X' if (c2 == '0' || ('3'..='9').contains(&c2)) => Some("RU"),
'Z' if ('A'..='R').contains(&c2) => Some("IT"),
_ => None,
}
}