use super::super::{sanitize, Verdict};
use anyhow::{anyhow, Result};
const WEIGHTS: [u32; 12] = [7, 6, 5, 4, 3, 2, 7, 6, 5, 4, 3, 2];
fn compute_check(body: &str) -> u32 {
let total: u32 = body
.chars()
.zip(WEIGHTS.iter())
.map(|(c, &w)| c.to_digit(10).unwrap() * w)
.sum();
let r = total % 11;
((11 - r) % 11) % 10
}
pub fn verify_mk_vat(input: &str) -> Verdict {
let clean = match super::strip_vat_prefix(input, "MK") {
Ok(body) => body,
Err(v) => return v,
};
if clean.len() != 13 {
return Verdict::Invalid {
reason: format!("MK EDB: expected 13 digits, got {}", clean.len()),
};
}
if !clean.chars().all(|c| c.is_ascii_digit()) {
return Verdict::Invalid { reason: "non-digit input".into() };
}
let body = &clean[..12];
let check: u32 = clean.chars().nth(12).unwrap().to_digit(10).unwrap();
let expected = compute_check(body);
if expected == check {
Verdict::Valid {
formatted: format!("MK{}", clean),
detected: "North Macedonian EDB".into(),
comment: String::new(),
}
} else {
Verdict::Invalid {
reason: format!("MK EDB check mismatch: expected {}, got {}", expected, check),
}
}
}
pub fn create_mk_vat(input: &str, _raw: bool) -> Result<String> {
let clean = sanitize(input, false);
if clean.len() != 12 {
return Err(anyhow!(
"MK EDB: expected 12 digits (body without check digit), got {}",
clean.len()
));
}
if !clean.chars().all(|c| c.is_ascii_digit()) {
return Err(anyhow!("non-digit input"));
}
let check = compute_check(&clean);
Ok(format!("MK{}{}", clean, check))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn mk_vat_valid_4030000375897() {
match verify_mk_vat("4030000375897") {
Verdict::Valid { formatted, detected, .. } => {
assert_eq!(formatted, "MK4030000375897");
assert_eq!(detected, "North Macedonian EDB");
}
v => panic!("{:?}", v),
}
}
#[test]
fn mk_vat_valid_4020990116747() {
match verify_mk_vat("4020990116747") {
Verdict::Valid { formatted, detected, .. } => {
assert_eq!(formatted, "MK4020990116747");
assert_eq!(detected, "North Macedonian EDB");
}
v => panic!("{:?}", v),
}
}
#[test]
fn mk_vat_valid_4057009501106() {
match verify_mk_vat("4057009501106") {
Verdict::Valid { formatted, detected, .. } => {
assert_eq!(formatted, "MK4057009501106");
assert_eq!(detected, "North Macedonian EDB");
}
v => panic!("{:?}", v),
}
}
#[test]
fn mk_vat_rejects_wrong_length() {
match verify_mk_vat("403000037589") {
Verdict::Invalid { reason } => assert!(reason.contains("expected 13 digits")),
v => panic!("{:?}", v),
}
}
#[test]
fn mk_vat_round_trip() {
let body = "403000037589";
let full = create_mk_vat(body, false).unwrap();
assert_eq!(full, "MK4030000375897");
match verify_mk_vat(&full) {
Verdict::Valid { .. } => {}
v => panic!("{:?}", v),
}
}
#[test]
fn mk_vat_rejects_bad_check() {
match verify_mk_vat("4030000375890") {
Verdict::Invalid { .. } => {}
v => panic!("{:?}", v),
}
}
}