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
|
|
|
|
2019-07-01 15:43:34 +00:00
|
|
|
use crate::common_types::{Class, Type, types};
|
|
|
|
use crate::errors::*;
|
|
|
|
use crate::records::structs;
|
|
|
|
use crate::ser::{RRData, StaticRRData};
|
|
|
|
use crate::ser::text::DnsTextContext;
|
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;
|
|
|
|
|
2019-07-01 15:43:34 +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> {
|
|
|
|
if name.len() >= TYPE_NAME_MAX_LEN { return None; }
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2019-09-08 13:34:28 +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>()
|
|
|
|
}
|
|
|
|
|
2019-09-08 13:34:28 +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> {
|
2019-09-08 13:34:28 +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>();
|
2017-12-21 12:32:14 +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>();
|
|
|
|
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>();
|
2017-12-16 20:58:18 +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>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_unknown("EID" , types::EID);
|
|
|
|
r.register_unknown("NIMLOC" , types::NIMLOC);
|
|
|
|
r.register_known::<structs::SRV>();
|
|
|
|
r.register_unknown("ATMA" , types::ATMA);
|
|
|
|
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>();
|
|
|
|
r.register_unknown("SINK" , types::SINK);
|
|
|
|
r.register_unknown("OPT" , types::OPT);
|
|
|
|
r.register_unknown("APL" , types::APL);
|
|
|
|
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>();
|
2017-12-16 20:58:18 +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>();
|
2017-12-16 20:58:18 +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>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_unknown("CSYNC" , types::CSYNC);
|
2019-07-02 15:37:40 +00:00
|
|
|
r.register_unknown("ZONEMD" , types::ZONEMD);
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_known::<structs::SPF>();
|
|
|
|
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>();
|
2017-12-16 20:58:18 +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>();
|
2017-12-16 20:58:18 +00:00
|
|
|
r.register_unknown("AVC" , types::AVC);
|
|
|
|
r.register_unknown("DOA" , types::DOA);
|
2019-07-02 15:37:40 +00:00
|
|
|
r.register_unknown("AMTRELAY" , types::AMTRELAY);
|
2017-12-26 16:30:08 +00:00
|
|
|
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();
|
2017-12-26 16:30:08 +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);
|
2017-12-26 16:30:08 +00:00
|
|
|
self.type_parser.insert(rrtype, Box::new(TagRRDataType::<T>(PhantomData)));
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_registration<T: StaticRRData + Sync + 'static>(&self) {
|
|
|
|
assert_eq!(self.names_to_type.get(T::NAME.as_bytes()), Some(&T::TYPE));
|
2019-09-08 13:34:28 +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
|
|
|
}
|
|
|
|
}
|