From d55bc78a353e8685924cbd6f896620dc3e4a4185 Mon Sep 17 00:00:00 2001 From: Javier Atenas Date: Mon, 6 Jan 2025 21:43:37 -0300 Subject: [PATCH] feat(zone.rs): added comments and a method to read a master file --- src/zones.rs | 273 ++++++++++++++++++++++++++++++++++++++++++++------- 1 file changed, 237 insertions(+), 36 deletions(-) diff --git a/src/zones.rs b/src/zones.rs index 8d32bdf6..6327eb77 100644 --- a/src/zones.rs +++ b/src/zones.rs @@ -1,42 +1,38 @@ +use std::fs::File; +use std::io::{self, BufRead}; +use std::path::Path; +pub mod SoaRdata; +/* +The following entries are defined: + [] + + $ORIGIN [] + + $INCLUDE [] [] + + [] + + [] + + contents take one of the following forms: + + [] [] + + [] [] +*/ + /// Structure to represent a DNS zone #[derive(Debug)] struct DnsZone { name: String, // Name of the zone (e.g., "example.com") ttl: u32, // Default time to live (in seconds) - soa: SoaRecord, // SOA (Start of Authority) record - ns_records: Vec,// List of name servers (NS) + soa: SoaRdata, // SOA (Start of Authority) record + ns_records: Vec,// List of name servers (NS) // Chech this part in the furure ******** a_records: Vec,// List of A records mx_records: Vec, // List of MX records other_records: Vec, // Other records (CNAME, PTR, etc.) } -/// Structure to represent an SOA record -#[derive(Debug)] -struct SoaRecord { - primary_ns: String, // Primary name server (MNAME) - admin_email: String, // Administrator's email (RNAME) - serial: u32, // Serial number - refresh: u32, // Refresh interval (in seconds) - retry: u32, // Retry interval (in seconds) - expire: u32, // Expiration time (in seconds) - minimum_ttl: u32, // Minimum TTL for zone records -} - -impl SoaRecord { - /// Create a new SOA record - fn new(primary_ns: &str, admin_email: &str, serial: u32, refresh: u32, retry: u32, expire: u32, minimum_ttl: u32) -> Self { - SoaRecord { - primary_ns: primary_ns.to_string(), - admin_email: admin_email.to_string(), - serial, - refresh, - retry, - expire, - minimum_ttl, - } - } -} - /// Structure for an A record (IPv4 address) #[derive(Debug)] struct ARecord { @@ -60,8 +56,29 @@ struct DnsRecord { } impl DnsZone { - /// Create a new DNS zone - fn new(name: &str, ttl: u32, soa: SoaRecord) -> Self { + /// Creates a new DNS zone. + /// + /// # Examples + /// + /// ``` + /// let soa = SoaRdata::new( + /// "ns1.example.com.", + /// "admin.example.com.", + /// 20240101, + /// 3600, + /// 1800, + /// 1209600, + /// 3600, + /// ); + /// + /// let dns_zone = DnsZone::new("example.com.", 3600, soa); + /// + /// assert_eq!(dns_zone.name, "example.com.".to_string()); + /// assert_eq!(dns_zone.ttl, 3600); + /// assert_eq!(dns_zone.soa.primary_ns, "ns1.example.com.".to_string()); + /// assert_eq!(dns_zone.soa.admin_email, "admin.example.com.".to_string()); + /// ``` + fn new(name: &str, ttl: u32, soa: SoaRdata) -> Self { DnsZone { name: name.to_string(), ttl, @@ -73,12 +90,43 @@ impl DnsZone { } } - /// Add an NS record + /// Adds a new NS (Name Server) record to the DNS zone. + /// + /// # Examples + /// + /// ``` + /// let mut dns_zone = DnsZone::new( + /// "example.com.", + /// 3600, + /// SoaRdata::new("ns1.example.com.", "admin.example.com.", 20240101, 3600, 1800, 1209600, 3600), + /// ); + /// + /// dns_zone.add_ns_record("ns2.example.com."); + /// + /// assert_eq!(dns_zone.ns_records.len(), 1); + /// assert_eq!(dns_zone.ns_records[0], "ns2.example.com.".to_string()); + /// ``` fn add_ns_record(&mut self, ns: &str) { self.ns_records.push(ns.to_string()); } - /// Add an A record + /// Adds a new A (Address) record to the DNS zone. + /// + /// # Examples + /// + /// ``` + /// let mut dns_zone = DnsZone::new( + /// "example.com.", + /// 3600, + /// SoaRdata::new("ns1.example.com.", "admin.example.com.", 20240101, 3600, 1800, 1209600, 3600), + /// ); + /// + /// dns_zone.add_a_record("www", "192.0.2.1"); + /// + /// assert_eq!(dns_zone.a_records.len(), 1); + /// assert_eq!(dns_zone.a_records[0].name, "www".to_string()); + /// assert_eq!(dns_zone.a_records[0].ip, "192.0.2.1".to_string()); + /// ``` fn add_a_record(&mut self, name: &str, ip: &str) { self.a_records.push(ARecord { name: name.to_string(), @@ -86,7 +134,24 @@ impl DnsZone { }); } - /// Add an MX record + /// Adds a new MX (Mail Exchange) record to the DNS zone. + /// + /// # Examples + /// + /// ``` + /// let mut dns_zone = DnsZone::new( + /// "example.com.", + /// 3600, + /// SoaRdata::new("ns1.example.com.", "admin.example.com.", 20240101, 3600, 1800, 1209600, 3600), + /// ); + /// + /// dns_zone.add_mx_record(10, "mail.example.com."); + /// + /// assert_eq!(dns_zone.mx_records.len(), 1); + /// assert_eq!(dns_zone.mx_records[0].priority, 10); + /// assert_eq!(dns_zone.mx_records[0].mail_server, "mail.example.com.".to_string()); + /// ``` + fn add_mx_record(&mut self, priority: u16, mail_server: &str) { self.mx_records.push(MxRecord { priority, @@ -94,7 +159,31 @@ impl DnsZone { }); } - /// Add a generic record + /// Adds a generic DNS record (e.g., CNAME, PTR, TXT) to the DNS zone. + /// + /// This method allows adding DNS records that do not have a dedicated function, + /// such as CNAME, PTR, or TXT records, by specifying their type, name, and value. + /// + /// # Examples + /// + /// ``` + /// let mut dns_zone = DnsZone::new( + /// "example.com.", + /// 3600, + /// SoaRdata::new("ns1.example.com.", "admin.example.com.", 20240101, 3600, 1800, 1209600, 3600), + /// ); + /// + /// dns_zone.add_generic_record("CNAME", "blog", "www.example.com."); + /// dns_zone.add_generic_record("TXT", "@", "v=spf1 include:example.com ~all"); + /// + /// assert_eq!(dns_zone.other_records.len(), 2); + /// assert_eq!(dns_zone.other_records[0].record_type, "CNAME".to_string()); + /// assert_eq!(dns_zone.other_records[0].name, "blog".to_string()); + /// assert_eq!(dns_zone.other_records[0].value, "www.example.com.".to_string()); + /// assert_eq!(dns_zone.other_records[1].record_type, "TXT".to_string()); + /// assert_eq!(dns_zone.other_records[1].name, "@".to_string()); + /// assert_eq!(dns_zone.other_records[1].value, "v=spf1 include:example.com ~all".to_string()); + /// ``` fn add_generic_record(&mut self, record_type: &str, name: &str, value: &str) { self.other_records.push(DnsRecord { record_type: record_type.to_string(), @@ -103,7 +192,119 @@ impl DnsZone { }); } - /// Get complete zone information + /// Create a new DNS zone from a master file + fn from_master_file(file_path: &str) -> io::Result { + // Open the master file + let path = Path::new(file_path); + let file = File::open(&path)?; + let reader = io::BufReader::new(file); + + let mut name = String::new(); + let mut ttl = 3600; // Default TTL + let mut soa = None; + let mut ns_records = Vec::new(); + let mut a_records = Vec::new(); + let mut mx_records = Vec::new(); + let mut other_records = Vec::new(); + + for line in reader.lines() { + let line = line?; + let line = line.trim(); + + // Ignore empty lines and comments + if line.is_empty() || line.starts_with(';') { + continue; + } + + // Process special directives + if line.starts_with("$ORIGIN") { + name = line.split_whitespace().nth(1).unwrap_or("").to_string(); + continue; + } + if line.starts_with("$TTL") { + ttl = line.split_whitespace().nth(1).unwrap_or("3600").parse().unwrap_or(3600); + continue; + } + + // Split line into fields + let parts: Vec<&str> = line.split_whitespace().collect(); + if parts.len() < 4 { + continue; // Malformed lines + } + + let record_name = parts[0]; + let record_type = parts[2]; + match record_type { + "SOA" => { + // Process SOA + if parts.len() >= 7 { + soa = Some(SoaRdata { + primary_ns: parts[3].to_string(), + admin_email: parts[4].to_string(), + serial: parts[5].parse().unwrap_or(0), + refresh: parts[6].parse().unwrap_or(3600), + retry: parts.get(7).unwrap_or(&"3600").parse().unwrap_or(3600), + expire: parts.get(8).unwrap_or(&"3600").parse().unwrap_or(1209600), + minimum_ttl: parts.get(9).unwrap_or(&"3600").parse().unwrap_or(3600), + }); + } + } + "NS" => { + // Process NS + ns_records.push(parts[3].to_string()); + } + "A" => { + // Process A + a_records.push(ARecord { + name: record_name.to_string(), + ip: parts[3].to_string(), + }); + } + "MX" => { + // Process MX + if parts.len() >= 5 { + mx_records.push(MxRecord { + priority: parts[3].parse().unwrap_or(10), + mail_server: parts[4].to_string(), + }); + } + } + _ => { + // Process other records (CNAME, PTR, etc.) + if parts.len() >= 4 { + other_records.push(DnsRecord { + record_type: record_type.to_string(), + name: record_name.to_string(), + value: parts[3].to_string(), + }); + } + } + } + } + } + + /// Returns a formatted string with all the DNS zone's information. + /// + /// # Examples + /// + /// ``` + /// let soa = SoaRdata::new( + /// "ns1.example.com.", + /// "admin.example.com.", + /// 20240101, + /// 3600, + /// 1800, + /// 1209600, + /// 3600, + /// ); + /// + /// let dns_zone = DnsZone::new("example.com.", 3600, soa); + /// + /// let info = dns_zone.get_info(); + /// assert!(info.contains("example.com.")); + /// assert!(info.contains("ns1.example.com.")); + /// assert!(info.contains("admin.example.com.")); + /// ``` fn get_info(&self) -> String { format!("{:#?}", self) }