use crate::carrier::Carrier;
use crate::consts;
use crate::country;
use crate::error;
use crate::extension::Extension;
use crate::is_viable;
use crate::metadata::{Database, DATABASE};
use crate::phone_number::{PhoneNumber, Type};
use crate::validator::{self, Validation};
use nom::{branch::alt, IResult};
#[macro_use]
pub mod helper;
pub mod natural;
pub mod rfc3966;
pub mod valid;
pub fn parse<S: AsRef<str>>(
country: Option<country::Id>,
string: S,
) -> Result<PhoneNumber, error::Parse> {
parse_with(&DATABASE, country, string)
}
pub fn parse_with<S: AsRef<str>>(
database: &Database,
country: Option<country::Id>,
string: S,
) -> Result<PhoneNumber, error::Parse> {
fn phone_number(i: &str) -> IResult<&str, helper::Number<'_>> {
parse! { i => alt((rfc3966::phone_number, natural::phone_number)) }
}
let (_, mut number) = phone_number(string.as_ref()).or(Err(error::Parse::NoNumber))?;
if !is_viable(&number.national) {
return Err(error::Parse::NoNumber);
}
number = helper::country_code(database, country, number)?;
let meta = if let Some(country) = &country {
database.by_id(country.as_ref())
} else {
let code = country::Code {
value: number.prefix.clone().map(|p| p.parse()).unwrap_or(Ok(0))?,
source: number.country,
};
use either::{Left, Right};
let without_zeros = number.national.trim_start_matches('0');
match validator::source_for(database, code.value(), without_zeros) {
Some(Left(region)) => database.by_id(region.as_ref()),
Some(Right(code)) => database.by_code(&code).and_then(|m| m.into_iter().next()),
None => None,
}
};
if let Some(meta) = meta {
let mut potential = helper::national_number(meta, number.clone());
if let Some(prefix) = meta.national_prefix.as_ref() {
if potential.national.starts_with(prefix) {
potential.national = helper::trim(potential.national, prefix.len());
}
}
if validator::length(meta, &potential, Type::Unknown) != Validation::TooShort {
number = potential;
}
}
if number.national.len() < consts::MIN_LENGTH_FOR_NSN {
return Err(error::Parse::TooShortNsn);
}
if number.national.len() > consts::MAX_LENGTH_FOR_NSN {
return Err(error::Parse::TooLong);
}
Ok(PhoneNumber {
code: country::Code {
value: number.prefix.map(|p| p.parse()).unwrap_or(Ok(0))?,
source: number.country,
},
national: number.national.parse()?,
extension: number.extension.map(|s| Extension(s.into_owned())),
carrier: number.carrier.map(|s| Carrier(s.into_owned())),
})
}
#[cfg(test)]
mod test {
use crate::country::{self, Source};
use crate::metadata::DATABASE;
use crate::national_number::NationalNumber;
use crate::parser;
use crate::phone_number::PhoneNumber;
use rstest::*;
#[rstest]
#[case(Source::Default, country::NZ, "033316005")]
#[case(Source::Default, country::NZ, "33316005")]
#[case(Source::Default, country::NZ, "03-331 6005")]
#[case(Source::Default, country::NZ, "03 331 6005")]
#[case(Source::Plus, country::NZ, "tel:03-331-6005;phone-context=+64")]
#[case(Source::Plus, country::NZ, "tel:03-331-6005;phone-context=+64;a=%A1")]
#[case(
Source::Plus,
country::NZ,
"tel:03-331-6005;isub=12345;phone-context=+64"
)]
#[case(Source::Plus, country::NZ, "tel:+64-3-331-6005;isub=12345")]
#[case(Source::Plus, country::NZ, "03-331-6005;phone-context=+64")]
#[case(Source::Idd, country::NZ, "0064 3 331 6005")]
#[case(Source::Idd, country::US, "01164 3 331 6005")]
#[case(Source::Plus, country::US, "+64 3 331 6005")]
#[case(Source::Plus, country::US, "+01164 3 331 6005")]
#[case(Source::Plus, country::NZ, "+0064 3 331 6005")]
#[case(Source::Plus, country::NZ, "+ 00 64 3 331 6005")]
fn parse_1(
#[case] source: Source,
#[case] country: impl Into<Option<country::Id>>,
#[case] number: &'static str,
) {
let reference = PhoneNumber {
code: country::Code { value: 64, source },
national: NationalNumber::new(33316005, 0).unwrap(),
extension: None,
carrier: None,
};
let country = country.into();
println!("parsing {} with country {:?}", number, country);
let parsed = parser::parse(country, number).unwrap();
println!("number type: {:?}", parsed.number_type(&DATABASE));
println!("parsed: {:?}", parsed);
assert_eq!(reference, parsed);
}
#[test]
fn parse_2() {
assert_eq!(
PhoneNumber {
code: country::Code {
value: 64,
source: Source::Number,
},
national: NationalNumber::new(64123456, 0).unwrap(),
extension: None,
carrier: None,
},
parser::parse(Some(country::NZ), "64(0)64123456").unwrap()
);
}
#[test]
fn parse_3() {
assert_eq!(
PhoneNumber {
code: country::Code {
value: 49,
source: country::Source::Default,
},
national: NationalNumber::new(30123456, 0).unwrap(),
extension: None,
carrier: None,
},
parser::parse(Some(country::DE), "301/23456").unwrap()
);
}
#[test]
fn parse_4() {
assert_eq!(
PhoneNumber {
code: country::Code {
value: 81,
source: country::Source::Plus,
},
national: NationalNumber::new(2345, 0,).unwrap(),
extension: None,
carrier: None,
},
parser::parse(Some(country::JP), "+81 *2345").unwrap()
);
}
#[test]
fn parse_5() {
assert_eq!(
PhoneNumber {
code: country::Code {
value: 64,
source: country::Source::Default,
},
national: NationalNumber::new(12, 0,).unwrap(),
extension: None,
carrier: None,
},
parser::parse(Some(country::NZ), "12").unwrap()
);
}
#[test]
fn parse_6() {
assert_eq!(
PhoneNumber {
code: country::Code {
value: 55,
source: country::Source::Default,
},
national: NationalNumber::new(3121286979, 0).unwrap(),
extension: None,
carrier: Some("12".into()),
},
parser::parse(Some(country::BR), "012 3121286979").unwrap()
);
}
#[test]
fn issue_43() {
let res = parser::parse(None, " 2 22#:");
assert!(res.is_err());
}
#[test]
fn advisory_1() {
let res = parser::parse(None, ".;phone-context=");
assert!(res.is_err(), "{res:?}");
}
#[test]
fn advisory_2() {
let res = parser::parse(None, "+dwPAA;phone-context=AA");
assert!(res.is_err(), "{res:?}");
}
#[test]
fn email() {
let res = parser::parse(Some(country::US), "someletters1110@gmail.com");
assert!(res.is_err(), "{res:?}");
}
}