use std::{cmp::Ordering, fmt, io, sync};
use thiserror::Error;
use tracing::debug;
use crate::proto::{
error::{ProtoError, ProtoErrorKind},
op::{Query, ResponseCode},
rr::{
rdata::SOA,
resource::{Record, RecordRef},
},
xfer::{retry_dns_handle::RetryableError, DnsResponse},
};
#[cfg(feature = "backtrace")]
use crate::proto::{trace, ExtBacktrace};
pub type ResolveResult<T> = ::std::result::Result<T, ResolveError>;
#[allow(clippy::large_enum_variant)]
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum ResolveErrorKind {
#[error("{0}")]
Message(&'static str),
#[error("{0}")]
Msg(String),
#[error("No connections available")]
NoConnections,
#[error("no record found for {:?}", query)]
NoRecordsFound {
query: Box<Query>,
soa: Option<Box<Record<SOA>>>,
negative_ttl: Option<u32>,
response_code: ResponseCode,
trusted: bool,
},
#[error("io error: {0}")]
Io(#[from] std::io::Error),
#[error("proto error: {0}")]
Proto(#[from] ProtoError),
#[error("request timed out")]
Timeout,
}
impl Clone for ResolveErrorKind {
fn clone(&self) -> Self {
use self::ResolveErrorKind::*;
match self {
NoConnections => NoConnections,
Message(msg) => Message(msg),
Msg(ref msg) => Msg(msg.clone()),
NoRecordsFound {
ref query,
ref soa,
negative_ttl,
response_code,
trusted,
} => NoRecordsFound {
query: query.clone(),
soa: soa.clone(),
negative_ttl: *negative_ttl,
response_code: *response_code,
trusted: *trusted,
},
Io(io) => Self::from(std::io::Error::from(io.kind())),
Proto(proto) => Self::from(proto.clone()),
Timeout => Timeout,
}
}
}
#[derive(Debug, Clone, Error)]
pub struct ResolveError {
pub(crate) kind: ResolveErrorKind,
#[cfg(feature = "backtrace")]
backtrack: Option<ExtBacktrace>,
}
impl ResolveError {
pub(crate) fn nx_error(
query: Query,
soa: Option<Record<SOA>>,
negative_ttl: Option<u32>,
response_code: ResponseCode,
trusted: bool,
) -> Self {
ResolveErrorKind::NoRecordsFound {
query: Box::new(query),
soa: soa.map(Box::new),
negative_ttl,
response_code,
trusted,
}
.into()
}
pub fn kind(&self) -> &ResolveErrorKind {
&self.kind
}
pub(crate) fn no_connections() -> Self {
Self {
kind: ResolveErrorKind::NoConnections,
#[cfg(feature = "backtrace")]
backtrack: trace!(),
}
}
pub(crate) fn is_no_connections(&self) -> bool {
matches!(self.kind, ResolveErrorKind::NoConnections)
}
pub fn from_response(response: DnsResponse, trust_nx: bool) -> Result<DnsResponse, Self> {
debug!("Response:{}", *response);
match response.response_code() {
response_code @ ResponseCode::ServFail
| response_code @ ResponseCode::Refused
| response_code @ ResponseCode::FormErr
| response_code @ ResponseCode::NotImp
| response_code @ ResponseCode::YXDomain
| response_code @ ResponseCode::YXRRSet
| response_code @ ResponseCode::NXRRSet
| response_code @ ResponseCode::NotAuth
| response_code @ ResponseCode::NotZone
| response_code @ ResponseCode::BADVERS
| response_code @ ResponseCode::BADSIG
| response_code @ ResponseCode::BADKEY
| response_code @ ResponseCode::BADTIME
| response_code @ ResponseCode::BADMODE
| response_code @ ResponseCode::BADNAME
| response_code @ ResponseCode::BADALG
| response_code @ ResponseCode::BADTRUNC
| response_code @ ResponseCode::BADCOOKIE => {
let response = response;
let soa = response.soa().as_ref().map(RecordRef::to_owned);
let query = response.queries().iter().next().cloned().unwrap_or_default();
let error_kind = ResolveErrorKind::NoRecordsFound {
query: Box::new(query),
soa: soa.map(Box::new),
negative_ttl: None,
response_code,
trusted: false,
};
Err(Self::from(error_kind))
}
response_code @ ResponseCode::NXDomain |
response_code @ ResponseCode::NoError
if !response.contains_answer() && !response.truncated() => {
let response = response;
let soa = response.soa().as_ref().map(RecordRef::to_owned);
let negative_ttl = response.negative_ttl();
let trusted = trust_nx && soa.is_some();
let query = response.into_message().take_queries().drain(..).next().unwrap_or_default();
let error_kind = ResolveErrorKind::NoRecordsFound {
query: Box::new(query),
soa: soa.map(Box::new),
negative_ttl,
response_code,
trusted,
};
Err(Self::from(error_kind))
}
ResponseCode::NXDomain
| ResponseCode::NoError
| ResponseCode::Unknown(_) => Ok(response),
}
}
pub(crate) fn cmp_specificity(&self, other: &Self) -> Ordering {
let kind = self.kind();
let other = other.kind();
match (kind, other) {
(ResolveErrorKind::NoRecordsFound { .. }, ResolveErrorKind::NoRecordsFound { .. }) => {
return Ordering::Equal
}
(ResolveErrorKind::NoRecordsFound { .. }, _) => return Ordering::Greater,
(_, ResolveErrorKind::NoRecordsFound { .. }) => return Ordering::Less,
_ => (),
}
match (kind, other) {
(ResolveErrorKind::Io { .. }, ResolveErrorKind::Io { .. }) => return Ordering::Equal,
(ResolveErrorKind::Io { .. }, _) => return Ordering::Greater,
(_, ResolveErrorKind::Io { .. }) => return Ordering::Less,
_ => (),
}
match (kind, other) {
(ResolveErrorKind::Proto { .. }, ResolveErrorKind::Proto { .. }) => {
return Ordering::Equal
}
(ResolveErrorKind::Proto { .. }, _) => return Ordering::Greater,
(_, ResolveErrorKind::Proto { .. }) => return Ordering::Less,
_ => (),
}
match (kind, other) {
(ResolveErrorKind::Timeout, ResolveErrorKind::Timeout) => return Ordering::Equal,
(ResolveErrorKind::Timeout, _) => return Ordering::Greater,
(_, ResolveErrorKind::Timeout) => return Ordering::Less,
_ => (),
}
Ordering::Equal
}
}
impl RetryableError for ResolveError {
fn should_retry(&self) -> bool {
match self.kind() {
ResolveErrorKind::Message(_)
| ResolveErrorKind::Msg(_)
| ResolveErrorKind::NoConnections
| ResolveErrorKind::NoRecordsFound { .. } => false,
ResolveErrorKind::Io(_) | ResolveErrorKind::Proto(_) | ResolveErrorKind::Timeout => {
true
}
}
}
fn attempted(&self) -> bool {
match self.kind() {
ResolveErrorKind::Proto(e) => !matches!(e.kind(), ProtoErrorKind::Busy),
_ => true,
}
}
}
impl fmt::Display for ResolveError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
cfg_if::cfg_if! {
if #[cfg(feature = "backtrace")] {
if let Some(ref backtrace) = self.backtrack {
fmt::Display::fmt(&self.kind, f)?;
fmt::Debug::fmt(backtrace, f)
} else {
fmt::Display::fmt(&self.kind, f)
}
} else {
fmt::Display::fmt(&self.kind, f)
}
}
}
}
impl From<ResolveErrorKind> for ResolveError {
fn from(kind: ResolveErrorKind) -> Self {
Self {
kind,
#[cfg(feature = "backtrace")]
backtrack: trace!(),
}
}
}
impl From<&'static str> for ResolveError {
fn from(msg: &'static str) -> Self {
ResolveErrorKind::Message(msg).into()
}
}
#[cfg(target_os = "windows")]
#[cfg(feature = "system-config")]
#[cfg_attr(docsrs, doc(cfg(all(feature = "system-config", windows))))]
impl From<ipconfig::error::Error> for ResolveError {
fn from(e: ipconfig::error::Error) -> ResolveError {
ResolveErrorKind::Msg(format!("failed to read from registry: {}", e)).into()
}
}
impl From<String> for ResolveError {
fn from(msg: String) -> Self {
ResolveErrorKind::Msg(msg).into()
}
}
impl From<io::Error> for ResolveError {
fn from(e: io::Error) -> Self {
match e.kind() {
io::ErrorKind::TimedOut => ResolveErrorKind::Timeout.into(),
_ => ResolveErrorKind::from(e).into(),
}
}
}
impl From<ProtoError> for ResolveError {
fn from(e: ProtoError) -> Self {
match *e.kind() {
ProtoErrorKind::Timeout => ResolveErrorKind::Timeout.into(),
_ => ResolveErrorKind::from(e).into(),
}
}
}
impl From<ResolveError> for io::Error {
fn from(e: ResolveError) -> Self {
match e.kind() {
ResolveErrorKind::Timeout => Self::new(io::ErrorKind::TimedOut, e),
_ => Self::new(io::ErrorKind::Other, e),
}
}
}
impl<T> From<sync::PoisonError<T>> for ResolveError {
fn from(e: sync::PoisonError<T>) -> Self {
ResolveErrorKind::Msg(format!("lock was poisoned, this is non-recoverable: {e}")).into()
}
}