rxing 0.8.5

A rust port of the zxing barcode library.
Documentation
/*
 * Copyright 2014 ZXing authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

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());

/**
 * Detects a result that is likely a vehicle identification number.
 *
 * @author Sean Owen
 */
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,
    }
}