use alloc::{
boxed::Box,
string::{String, ToString},
};
use core::{fmt, str::FromStr};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use crate::{
error::ProtoResult,
rr::{RData, RecordData, RecordType, domain::Name},
serialize::{binary::*, txt::ParseError},
};
#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
#[derive(Debug, PartialEq, Eq, Hash, Clone)]
#[non_exhaustive]
pub struct NAPTR {
pub order: u16,
pub preference: u16,
pub flags: Box<[u8]>,
pub services: Box<[u8]>,
pub regexp: Box<[u8]>,
pub replacement: Name,
}
impl NAPTR {
pub fn new(
order: u16,
preference: u16,
flags: Box<[u8]>,
services: Box<[u8]>,
regexp: Box<[u8]>,
replacement: Name,
) -> Self {
Self {
order,
preference,
flags,
services,
regexp,
replacement,
}
}
pub(crate) fn from_tokens<'i, I: Iterator<Item = &'i str>>(
mut tokens: I,
origin: Option<&Name>,
) -> Result<Self, ParseError> {
let order: u16 = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("order".to_string()))
.and_then(|s| u16::from_str(s).map_err(Into::into))?;
let preference: u16 = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("preference".to_string()))
.and_then(|s| u16::from_str(s).map_err(Into::into))?;
let flags = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("flags".to_string()))
.map(ToString::to_string)
.map(|s| s.into_bytes().into_boxed_slice())?;
if !verify_flags(&flags) {
return Err(ParseError::from("bad flags, must be in range [a-zA-Z0-9]"));
}
let service = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("service".to_string()))
.map(ToString::to_string)
.map(|s| s.into_bytes().into_boxed_slice())?;
let regexp = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("regexp".to_string()))
.map(ToString::to_string)
.map(|s| s.into_bytes().into_boxed_slice())?;
let replacement: Name = tokens
.next()
.ok_or_else(|| ParseError::MissingToken("replacement".to_string()))
.and_then(|s| Name::parse(s, origin).map_err(ParseError::from))?;
Ok(Self::new(
order,
preference,
flags,
service,
regexp,
replacement,
))
}
}
pub fn verify_flags(flags: &[u8]) -> bool {
flags
.iter()
.all(|c| matches!(c, b'0'..=b'9' | b'a'..=b'z' | b'A'..=b'Z'))
}
impl BinEncodable for NAPTR {
fn emit(&self, encoder: &mut BinEncoder<'_>) -> ProtoResult<()> {
let mut encoder = encoder.with_rdata_behavior(RDataEncoding::Canonical);
self.order.emit(&mut encoder)?;
self.preference.emit(&mut encoder)?;
encoder.emit_character_data(&self.flags)?;
encoder.emit_character_data(&self.services)?;
encoder.emit_character_data(&self.regexp)?;
self.replacement.emit(&mut encoder)?;
Ok(())
}
}
impl<'r> BinDecodable<'r> for NAPTR {
fn read(decoder: &mut BinDecoder<'r>) -> Result<Self, DecodeError> {
Ok(Self::new(
decoder.read_u16()?.unverified(),
decoder.read_u16()?.unverified(),
decoder
.read_character_data()?
.verify_unwrap(|s| verify_flags(s))
.map_err(|_| DecodeError::NaptrFlagsInvalid)?
.to_vec()
.into_boxed_slice(),
decoder.read_character_data()?.unverified().to_vec().into_boxed_slice(),
decoder.read_character_data()?.unverified().to_vec().into_boxed_slice(),
Name::read(decoder)?,
))
}
}
impl RecordData for NAPTR {
fn try_borrow(data: &RData) -> Option<&Self> {
match data {
RData::NAPTR(csync) => Some(csync),
_ => None,
}
}
fn record_type(&self) -> RecordType {
RecordType::NAPTR
}
fn into_rdata(self) -> RData {
RData::NAPTR(self)
}
}
impl fmt::Display for NAPTR {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> Result<(), fmt::Error> {
write!(
f,
"{order} {pref} \"{flags}\" \"{service}\" \"{regexp}\" {replace}",
order = self.order,
pref = self.preference,
flags = &String::from_utf8_lossy(&self.flags),
service = &String::from_utf8_lossy(&self.services),
regexp = &String::from_utf8_lossy(&self.regexp),
replace = self.replacement
)
}
}
#[cfg(test)]
mod tests {
#![allow(clippy::dbg_macro, clippy::print_stdout)]
use alloc::vec::Vec;
#[cfg(feature = "std")]
use std::println;
use super::*;
#[test]
fn test() {
use core::str::FromStr;
let rdata = NAPTR::new(
8,
16,
b"aa11AA".to_vec().into_boxed_slice(),
b"services".to_vec().into_boxed_slice(),
b"regexpr".to_vec().into_boxed_slice(),
Name::from_str("naptr.example.com.").unwrap(),
);
let mut bytes = Vec::new();
let mut encoder: BinEncoder<'_> = BinEncoder::new(&mut bytes);
assert!(rdata.emit(&mut encoder).is_ok());
let bytes = encoder.into_bytes();
#[cfg(feature = "std")]
println!("bytes: {bytes:?}");
let mut decoder: BinDecoder<'_> = BinDecoder::new(bytes);
let read_rdata = NAPTR::read(&mut decoder).expect("Decoding error");
assert_eq!(rdata, read_rdata);
}
#[test]
fn test_bad_data() {
use core::str::FromStr;
let rdata = NAPTR::new(
8,
16,
b"aa11AA-".to_vec().into_boxed_slice(),
b"services".to_vec().into_boxed_slice(),
b"regexpr".to_vec().into_boxed_slice(),
Name::from_str("naptr.example.com").unwrap(),
);
let mut bytes = Vec::new();
let mut encoder: BinEncoder<'_> = BinEncoder::new(&mut bytes);
assert!(rdata.emit(&mut encoder).is_ok());
let bytes = encoder.into_bytes();
#[cfg(feature = "std")]
println!("bytes: {bytes:?}");
let mut decoder: BinDecoder<'_> = BinDecoder::new(bytes);
let read_rdata = NAPTR::read(&mut decoder);
assert!(
read_rdata.is_err(),
"should have failed decoding with bad flag data"
);
}
#[test]
fn test_parsing() {
assert_eq!(
NAPTR::from_tokens(
vec!["100", "50", "a", "z3950+N2L+N2C", "", "cidserver"].into_iter(),
Some(&Name::from_str("example.com.").unwrap())
)
.expect("failed to parse NAPTR"),
NAPTR::new(
100,
50,
b"a".to_vec().into_boxed_slice(),
b"z3950+N2L+N2C".to_vec().into_boxed_slice(),
b"".to_vec().into_boxed_slice(),
Name::from_str("cidserver.example.com.").unwrap()
),
);
}
#[test]
fn test_parsing_fails() {
assert!(
NAPTR::from_tokens(
vec!["100", "50", "-", "z3950+N2L+N2C", "", "cidserver"].into_iter(),
Some(&Name::from_str("example.com.").unwrap())
)
.is_err()
);
}
}