2017-12-26 16:30:08 +00:00
|
|
|
use bytes::Bytes;
|
|
|
|
use std::any::TypeId;
|
|
|
|
use std::collections::HashMap;
|
|
|
|
use std::io::Cursor;
|
2017-12-21 12:32:14 +00:00
|
|
|
use std::marker::PhantomData;
|
2017-12-16 20:58:18 +00:00
|
|
|
|
2020-03-07 15:57:47 +00:00
|
|
|
use crate::common_types::{types, Class, Type};
|
2019-07-01 15:43:34 +00:00
|
|
|
use crate::errors::*;
|
|
|
|
use crate::records::structs;
|
|
|
|
use crate::ser::text::DnsTextContext;
|
2020-03-07 15:57:47 +00:00
|
|
|
use crate::ser::{RRData, StaticRRData};
|
2017-12-16 20:58:18 +00:00
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
// this should be enough for registered names
|
|
|
|
const TYPE_NAME_MAX_LEN: usize = 16;
|
|
|
|
|
2020-03-07 15:57:47 +00:00
|
|
|
lazy_static::lazy_static! {
|
2017-12-16 20:58:18 +00:00
|
|
|
static ref REGISTRY: Registry = Registry::init();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn registry() -> &'static Registry {
|
|
|
|
&*REGISTRY
|
|
|
|
}
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
pub(crate) fn lookup_type_name(name: &str) -> Option<Type> {
|
2020-03-07 15:57:47 +00:00
|
|
|
if name.len() >= TYPE_NAME_MAX_LEN {
|
|
|
|
return None;
|
|
|
|
}
|
2017-12-26 16:30:08 +00:00
|
|
|
let mut name_buf_storage = [0u8; TYPE_NAME_MAX_LEN];
|
|
|
|
let name_buf = &mut name_buf_storage[..name.len()];
|
|
|
|
name_buf.copy_from_slice(name.as_bytes());
|
2017-12-16 20:58:18 +00:00
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
name_buf.make_ascii_uppercase();
|
|
|
|
|
|
|
|
let registry = registry();
|
|
|
|
let &t = registry.names_to_type.get(name_buf)?;
|
2017-12-16 20:58:18 +00:00
|
|
|
Some(t)
|
|
|
|
}
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
pub fn known_name_to_type(name: &str) -> Option<Type> {
|
2017-12-16 20:58:18 +00:00
|
|
|
let registry = registry();
|
2017-12-26 16:30:08 +00:00
|
|
|
let t = lookup_type_name(name)?;
|
|
|
|
registry.type_parser.get(&t)?;
|
|
|
|
|
2017-12-16 20:58:18 +00:00
|
|
|
Some(t)
|
|
|
|
}
|
|
|
|
|
2020-03-07 15:57:47 +00:00
|
|
|
pub fn deserialize_rr_data(
|
|
|
|
ttl: u32,
|
|
|
|
rr_class: Class,
|
|
|
|
rr_type: Type,
|
|
|
|
data: &mut Cursor<Bytes>,
|
|
|
|
) -> Result<Box<dyn RRData>> {
|
2017-12-27 11:41:54 +00:00
|
|
|
let registry = registry();
|
|
|
|
match registry.type_parser.get(&rr_type) {
|
|
|
|
Some(p) => p.deserialize_rr_data(ttl, rr_class, rr_type, data),
|
|
|
|
None => Ok(Box::new(super::UnknownRecord::deserialize(rr_type, data)?) as _),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-08 13:34:28 +00:00
|
|
|
pub fn parse_rr_data(context: &DnsTextContext, data: &mut &str) -> Result<Box<dyn RRData>> {
|
2017-12-27 11:41:54 +00:00
|
|
|
let registry = registry();
|
|
|
|
let t = match context.record_type() {
|
|
|
|
Some(t) => t,
|
2019-07-01 15:43:34 +00:00
|
|
|
None => failure::bail!("require record type to parse record data"),
|
2017-12-27 11:41:54 +00:00
|
|
|
};
|
|
|
|
match registry.type_parser.get(&t) {
|
|
|
|
Some(p) => p.parse_rr_data(context, data),
|
2019-07-01 15:43:34 +00:00
|
|
|
None => failure::bail!("unknown type: {}", t),
|
2017-12-27 11:41:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
pub(crate) fn lookup_type_to_name(rrtype: Type) -> Option<&'static str> {
|
2017-12-21 12:32:14 +00:00
|
|
|
let registry = registry();
|
|
|
|
registry.type_names.get(&rrtype).map(|s| s as _)
|
|
|
|
}
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
#[doc(hidden)]
|
|
|
|
pub fn check_registration<T: StaticRRData + Sync + 'static>() {
|
|
|
|
registry().check_registration::<T>();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
2017-12-21 12:32:14 +00:00
|
|
|
struct TagRRDataType<T: RRData>(PhantomData<T>);
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
trait RRDataTypeParse: 'static {
|
|
|
|
fn type_id(&self) -> TypeId {
|
|
|
|
TypeId::of::<Self>()
|
|
|
|
}
|
|
|
|
|
2020-03-07 15:57:47 +00:00
|
|
|
fn deserialize_rr_data(
|
|
|
|
&self,
|
|
|
|
ttl: u32,
|
|
|
|
rr_class: Class,
|
|
|
|
rr_type: Type,
|
|
|
|
data: &mut Cursor<Bytes>,
|
|
|
|
) -> Result<Box<dyn RRData>>;
|
2017-12-27 11:41:54 +00:00
|
|
|
|
2019-09-08 13:34:28 +00:00
|
|
|
fn parse_rr_data(&self, context: &DnsTextContext, data: &mut &str) -> Result<Box<dyn RRData>>;
|
2017-12-26 16:30:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: RRData + 'static> RRDataTypeParse for TagRRDataType<T> {
|
2020-03-07 15:57:47 +00:00
|
|
|
fn deserialize_rr_data(
|
|
|
|
&self,
|
|
|
|
ttl: u32,
|
|
|
|
rr_class: Class,
|
|
|
|
rr_type: Type,
|
|
|
|
data: &mut Cursor<Bytes>,
|
|
|
|
) -> Result<Box<dyn RRData>> {
|
2017-12-26 16:30:08 +00:00
|
|
|
T::deserialize_rr_data(ttl, rr_class, rr_type, data).map(|d| Box::new(d) as _)
|
|
|
|
}
|
2017-12-27 11:41:54 +00:00
|
|
|
|
2019-09-08 13:34:28 +00:00
|
|
|
fn parse_rr_data(&self, context: &DnsTextContext, data: &mut &str) -> Result<Box<dyn RRData>> {
|
2017-12-27 11:41:54 +00:00
|
|
|
T::dns_parse_rr_data(context, data).map(|d| Box::new(d) as _)
|
|
|
|
}
|
2017-12-26 16:30:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-16 20:58:18 +00:00
|
|
|
struct Registry {
|
2017-12-26 16:30:08 +00:00
|
|
|
// store (ascii) upper-case names.
|
|
|
|
names_to_type: HashMap<Vec<u8>, Type>,
|
2017-12-21 12:32:14 +00:00
|
|
|
type_names: HashMap<Type, String>,
|
2019-09-08 13:34:28 +00:00
|
|
|
type_parser: HashMap<Type, Box<dyn RRDataTypeParse + Sync>>,
|
2017-12-16 20:58:18 +00:00
|
|
|
// make sure registrations are in order
|
|
|
|
prev_type: Option<Type>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Registry {
|
|
|
|
fn init() -> Self {
|
|
|
|
let mut r = Registry {
|
|
|
|
names_to_type: HashMap::new(),
|
2017-12-21 12:32:14 +00:00
|
|
|
type_names: HashMap::new(),
|
2017-12-16 20:58:18 +00:00
|
|
|
type_parser: HashMap::new(),
|
|
|
|
prev_type: None,
|
|
|
|
};
|
|
|
|
|
|
|
|
r.register_known::<structs::A>();
|
|
|
|
r.register_known::<structs::NS>();
|
|
|
|
r.register_known::<structs::MD>();
|
|
|
|
r.register_known::<structs::MF>();
|
|
|
|
r.register_known::<structs::CNAME>();
|
|
|
|
r.register_known::<structs::SOA>();
|
|
|
|
r.register_known::<structs::MB>();
|
|
|
|
r.register_known::<structs::MG>();
|
|
|
|
r.register_known::<structs::MR>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("NULL", types::NULL);
|
|
|
|
r.register_unknown("WKS", types::WKS);
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::PTR>();
|
|
|
|
r.register_known::<structs::HINFO>();
|
|
|
|
r.register_known::<structs::MINFO>();
|
|
|
|
r.register_known::<structs::MX>();
|
|
|
|
r.register_known::<structs::TXT>();
|
|
|
|
r.register_known::<structs::RP>();
|
|
|
|
r.register_known::<structs::AFSDB>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("X25", types::X25);
|
|
|
|
r.register_unknown("ISDN", types::ISDN);
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::RT>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("NSAP", types::NSAP);
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::NSAP_PTR>();
|
|
|
|
r.register_known::<structs::SIG>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::KEY>();
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::PX>();
|
2017-12-26 16:30:08 +00:00
|
|
|
r.register_known::<structs::GPOS>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::AAAA>();
|
|
|
|
r.register_known::<structs::LOC>();
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::NXT>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("EID", types::EID);
|
|
|
|
r.register_unknown("NIMLOC", types::NIMLOC);
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::SRV>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("ATMA", types::ATMA);
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::NAPTR>();
|
|
|
|
r.register_known::<structs::KX>();
|
|
|
|
r.register_known::<structs::CERT>();
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::A6>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::DNAME>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("SINK", types::SINK);
|
|
|
|
r.register_unknown("OPT", types::OPT);
|
2020-03-07 15:11:28 +00:00
|
|
|
r.register_known::<structs::APL>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::DS>();
|
|
|
|
r.register_known::<structs::SSHFP>();
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::IPSECKEY>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::RRSIG>();
|
|
|
|
r.register_known::<structs::NSEC>();
|
|
|
|
r.register_known::<structs::DNSKEY>();
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::DHCID>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::NSEC3>();
|
|
|
|
r.register_known::<structs::NSEC3PARAM>();
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::TLSA>();
|
|
|
|
r.register_known::<structs::SMIMEA>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("HIP", types::HIP);
|
2017-12-29 13:07:58 +00:00
|
|
|
r.register_known::<structs::NINFO>();
|
|
|
|
r.register_known::<structs::RKEY>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("TALINK", types::TALINK);
|
2017-12-29 13:07:58 +00:00
|
|
|
r.register_known::<structs::CDS>();
|
|
|
|
r.register_known::<structs::CDNSKEY>();
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::OPENPGPKEY>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("CSYNC", types::CSYNC);
|
|
|
|
r.register_unknown("ZONEMD", types::ZONEMD);
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::SPF>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("UINFO", types::UINFO);
|
|
|
|
r.register_unknown("UID", types::UID);
|
|
|
|
r.register_unknown("GID", types::GID);
|
|
|
|
r.register_unknown("UNSPEC", types::UNSPEC);
|
|
|
|
r.register_unknown("NID", types::NID);
|
|
|
|
r.register_unknown("L32", types::L32);
|
|
|
|
r.register_unknown("L64", types::L64);
|
|
|
|
r.register_unknown("LP", types::LP);
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::EUI48>();
|
|
|
|
r.register_known::<structs::EUI64>();
|
|
|
|
r.register_known::<structs::TKEY>();
|
|
|
|
r.register_known::<structs::TSIG>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("IXFR", types::IXFR);
|
|
|
|
r.register_unknown("AXFR", types::AXFR);
|
|
|
|
r.register_unknown("MAILB", types::MAILB);
|
|
|
|
r.register_unknown("MAILA", types::MAILA);
|
|
|
|
r.register_unknown("ANY", types::ANY);
|
2017-12-21 12:32:14 +00:00
|
|
|
r.register_known::<structs::URI>();
|
|
|
|
r.register_known::<structs::CAA>();
|
2020-03-07 15:57:47 +00:00
|
|
|
r.register_unknown("AVC", types::AVC);
|
|
|
|
r.register_unknown("DOA", types::DOA);
|
|
|
|
r.register_unknown("AMTRELAY", types::AMTRELAY);
|
|
|
|
r.register_unknown("TA", types::TA);
|
2017-12-27 20:50:51 +00:00
|
|
|
r.register_known::<structs::DLV>();
|
2017-12-26 16:30:08 +00:00
|
|
|
r.register_known::<structs::ALIAS>();
|
|
|
|
|
|
|
|
// "ALL" could be an alias for the ANY type?
|
|
|
|
// assert!(r.names_to_type.insert("ALL".into(), types::ANY).is_none());
|
2017-12-16 20:58:18 +00:00
|
|
|
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
2017-12-21 12:32:14 +00:00
|
|
|
fn register_name(&mut self, name: &str, rrtype: Type) {
|
2017-12-16 20:58:18 +00:00
|
|
|
assert!(self.prev_type < Some(rrtype), "registration not in order");
|
|
|
|
self.prev_type = Some(rrtype);
|
2017-12-21 12:32:14 +00:00
|
|
|
let mut name: String = name.into();
|
|
|
|
name.make_ascii_uppercase();
|
2020-03-07 15:57:47 +00:00
|
|
|
assert!(
|
|
|
|
!name.starts_with("TYPE"),
|
|
|
|
"must not register generic name: {}",
|
|
|
|
name
|
|
|
|
);
|
|
|
|
assert!(
|
|
|
|
name.len() <= TYPE_NAME_MAX_LEN,
|
|
|
|
"name too long: {} - maybe you need to increase TYPE_NAME_MAX_LEN",
|
|
|
|
name
|
|
|
|
);
|
|
|
|
assert!(self
|
|
|
|
.names_to_type
|
|
|
|
.insert(name.clone().into_bytes(), rrtype)
|
|
|
|
.is_none());
|
2017-12-21 12:32:14 +00:00
|
|
|
self.type_names.insert(rrtype, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn register_unknown(&mut self, name: &'static str, rrtype: Type) {
|
|
|
|
self.register_name(name, rrtype);
|
2017-12-16 20:58:18 +00:00
|
|
|
}
|
|
|
|
|
2017-12-26 16:30:08 +00:00
|
|
|
fn register_known<T: StaticRRData + Sync + 'static>(&mut self) {
|
2017-12-21 12:32:14 +00:00
|
|
|
let rrtype = T::TYPE;
|
|
|
|
let name = T::NAME;
|
|
|
|
self.register_name(name, rrtype);
|
2020-03-07 15:57:47 +00:00
|
|
|
self.type_parser
|
|
|
|
.insert(rrtype, Box::new(TagRRDataType::<T>(PhantomData)));
|
2017-12-26 16:30:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn check_registration<T: StaticRRData + Sync + 'static>(&self) {
|
|
|
|
assert_eq!(self.names_to_type.get(T::NAME.as_bytes()), Some(&T::TYPE));
|
2020-03-07 15:57:47 +00:00
|
|
|
let p: &dyn RRDataTypeParse = &**self
|
|
|
|
.type_parser
|
|
|
|
.get(&T::TYPE)
|
|
|
|
.expect("no parser registered");
|
2017-12-26 16:30:08 +00:00
|
|
|
let tid = TypeId::of::<TagRRDataType<T>>();
|
|
|
|
assert_eq!(p.type_id(), tid);
|
2017-12-16 20:58:18 +00:00
|
|
|
}
|
|
|
|
}
|