use serde::{Deserialize, Serialize};
use tracing::{debug, instrument};
use crate::dns::{DnsRecord, DnsResolver, RecordType};
use crate::error::Result;
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ServerResult {
pub nameserver: String,
pub records: Vec<DnsRecord>,
pub error: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct DnsComparison {
pub domain: String,
pub record_type: RecordType,
pub server_a: ServerResult,
pub server_b: ServerResult,
pub matches: bool,
pub only_in_a: Vec<String>,
pub only_in_b: Vec<String>,
pub common: Vec<String>,
}
pub struct DnsComparator {
resolver: DnsResolver,
}
impl Default for DnsComparator {
fn default() -> Self {
Self::new()
}
}
impl DnsComparator {
pub fn new() -> Self {
Self {
resolver: DnsResolver::new(),
}
}
#[instrument(skip(self), fields(domain = %domain, record_type = %record_type, server_a = %server_a, server_b = %server_b))]
pub async fn compare(
&self,
domain: &str,
record_type: RecordType,
server_a: &str,
server_b: &str,
) -> Result<DnsComparison> {
let domain = crate::validation::normalize_domain(domain)?;
let (result_a, result_b) = tokio::join!(
self.resolver.resolve(&domain, record_type, Some(server_a)),
self.resolver.resolve(&domain, record_type, Some(server_b))
);
let server_a_result = match result_a {
Ok(records) => ServerResult {
nameserver: server_a.to_string(),
records,
error: None,
},
Err(e) => ServerResult {
nameserver: server_a.to_string(),
records: vec![],
error: Some(e.to_string()),
},
};
let server_b_result = match result_b {
Ok(records) => ServerResult {
nameserver: server_b.to_string(),
records,
error: None,
},
Err(e) => ServerResult {
nameserver: server_b.to_string(),
records: vec![],
error: Some(e.to_string()),
},
};
let values_a: std::collections::HashSet<String> = server_a_result
.records
.iter()
.map(|r| r.format_short())
.collect();
let values_b: std::collections::HashSet<String> = server_b_result
.records
.iter()
.map(|r| r.format_short())
.collect();
let mut only_in_a: Vec<String> = values_a.difference(&values_b).cloned().collect();
let mut only_in_b: Vec<String> = values_b.difference(&values_a).cloned().collect();
let mut common: Vec<String> = values_a.intersection(&values_b).cloned().collect();
only_in_a.sort();
only_in_b.sort();
common.sort();
let matches = only_in_a.is_empty()
&& only_in_b.is_empty()
&& server_a_result.error.is_none()
&& server_b_result.error.is_none();
debug!(
matches = matches,
common = common.len(),
only_in_a = only_in_a.len(),
only_in_b = only_in_b.len(),
"DNS comparison complete"
);
Ok(DnsComparison {
domain: domain.to_string(),
record_type,
server_a: server_a_result,
server_b: server_b_result,
matches,
only_in_a,
only_in_b,
common,
})
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::dns::{RecordData, RecordType};
#[test]
fn test_dns_comparison_serialization() {
let comparison = DnsComparison {
domain: "example.com".to_string(),
record_type: RecordType::A,
server_a: ServerResult {
nameserver: "8.8.8.8".to_string(),
records: vec![DnsRecord {
name: "example.com".to_string(),
record_type: RecordType::A,
ttl: 300,
data: RecordData::A {
address: "93.184.216.34".to_string(),
},
}],
error: None,
},
server_b: ServerResult {
nameserver: "1.1.1.1".to_string(),
records: vec![DnsRecord {
name: "example.com".to_string(),
record_type: RecordType::A,
ttl: 300,
data: RecordData::A {
address: "93.184.216.34".to_string(),
},
}],
error: None,
},
matches: true,
only_in_a: vec![],
only_in_b: vec![],
common: vec!["93.184.216.34".to_string()],
};
let json = serde_json::to_string(&comparison).unwrap();
assert!(json.contains("example.com"));
assert!(json.contains("93.184.216.34"));
assert!(json.contains("\"matches\":true"));
}
#[test]
fn test_server_result_with_error() {
let result = ServerResult {
nameserver: "8.8.8.8".to_string(),
records: vec![],
error: Some("connection timed out".to_string()),
};
let json = serde_json::to_string(&result).unwrap();
assert!(json.contains("connection timed out"));
}
}