Move flags to config file, and add more options. Mirror some in the module.
187 lines
5.5 KiB
Rust
187 lines
5.5 KiB
Rust
//! Deal with the DNS records
|
|
|
|
use miette::{ensure, miette, LabeledSpan, Result};
|
|
|
|
pub fn validate_record_str(record: &str) -> Result<()> {
|
|
validate_line(0, record).map_err(|err| err.with_source_code(String::from(record)))
|
|
}
|
|
|
|
fn validate_line(offset: usize, line: &str) -> Result<()> {
|
|
if line.is_empty() {
|
|
return Ok(());
|
|
}
|
|
|
|
ensure!(
|
|
line.len() <= 255,
|
|
miette!(
|
|
labels = [LabeledSpan::new(
|
|
Some("this line".to_string()),
|
|
offset,
|
|
line.len(),
|
|
)],
|
|
help = "fully qualified domain names can be at most 255 characters long",
|
|
url = "https://en.wikipedia.org/wiki/Fully_qualified_domain_name",
|
|
"hostname too long ({} octets)",
|
|
line.len(),
|
|
)
|
|
);
|
|
ensure!(
|
|
line.ends_with('.'),
|
|
miette!(
|
|
labels = [LabeledSpan::new(
|
|
Some("last character".to_string()),
|
|
offset + line.len() - 1,
|
|
1,
|
|
)],
|
|
help = "hostname should be a fully qualified domain name (end with a '.')",
|
|
url = "https://en.wikipedia.org/wiki/Fully_qualified_domain_name",
|
|
"not a fully qualified domain name"
|
|
)
|
|
);
|
|
|
|
let mut label_offset = 0usize;
|
|
for label in line.strip_suffix('.').unwrap_or(line).split('.') {
|
|
validate_label(offset + label_offset, label)?;
|
|
label_offset += label.len() + 1;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn validate_label(offset: usize, label: &str) -> Result<()> {
|
|
ensure!(
|
|
!label.is_empty(),
|
|
miette!(
|
|
labels = [LabeledSpan::new(
|
|
Some("label".to_string()),
|
|
offset,
|
|
label.len(),
|
|
)],
|
|
help = "each label should have at least one character",
|
|
url = "https://en.wikipedia.org/wiki/Fully_qualified_domain_name",
|
|
"empty label",
|
|
)
|
|
);
|
|
ensure!(
|
|
label.len() <= 63,
|
|
miette!(
|
|
labels = [LabeledSpan::new(
|
|
Some("label".to_string()),
|
|
offset,
|
|
label.len(),
|
|
)],
|
|
help = "labels should be at most 63 octets",
|
|
url = "https://en.wikipedia.org/wiki/Fully_qualified_domain_name",
|
|
"label too long ({} octets)",
|
|
label.len(),
|
|
)
|
|
);
|
|
|
|
for (octet_offset, octet) in label.bytes().enumerate() {
|
|
validate_octet(offset + octet_offset, octet)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn validate_octet(offset: usize, octet: u8) -> Result<()> {
|
|
let spans = || [LabeledSpan::new(Some("octet".to_string()), offset, 1)];
|
|
ensure!(
|
|
octet.is_ascii(),
|
|
miette!(
|
|
labels = spans(),
|
|
help = "we only accept ascii characters",
|
|
url = "https://en.wikipedia.org/wiki/Hostname#Syntax",
|
|
"invalid octet: '{}'",
|
|
octet.escape_ascii(),
|
|
)
|
|
);
|
|
|
|
ensure!(
|
|
octet.is_ascii_alphanumeric() || octet == b'-' || octet == b'_',
|
|
miette!(
|
|
labels = spans(),
|
|
help = "hostnames are only allowed to contain characters in [a-zA-Z0-9_-]",
|
|
url = "https://en.wikipedia.org/wiki/Hostname#Syntax",
|
|
"invalid octet: '{}'",
|
|
octet.escape_ascii(),
|
|
)
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use crate::records::validate_record_str;
|
|
|
|
macro_rules! assert_miette_snapshot {
|
|
($diag:expr) => {{
|
|
use std::borrow::Borrow;
|
|
|
|
use insta::{with_settings, assert_snapshot};
|
|
use miette::{GraphicalReportHandler, GraphicalTheme};
|
|
|
|
let mut out = String::new();
|
|
GraphicalReportHandler::new_themed(GraphicalTheme::unicode_nocolor())
|
|
.with_width(80)
|
|
.render_report(&mut out, $diag.borrow())
|
|
.unwrap();
|
|
with_settings!({
|
|
description => stringify!($diag)
|
|
}, {
|
|
assert_snapshot!(out);
|
|
});
|
|
}};
|
|
}
|
|
|
|
#[test]
|
|
fn valid_records() -> miette::Result<()> {
|
|
for record in [
|
|
"example.com.",
|
|
"example.org.",
|
|
"example.net.",
|
|
"subdomain.example.com.",
|
|
] {
|
|
validate_record_str(record)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn hostname_too_long() {
|
|
let err = validate_record_str("example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.example.net.").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
|
|
#[test]
|
|
fn not_fqd() {
|
|
let err = validate_record_str("example.net").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
|
|
#[test]
|
|
fn empty_label() {
|
|
let err = validate_record_str("name..example.org.").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
|
|
#[test]
|
|
fn label_too_long() {
|
|
let err = validate_record_str("name.an-entremely-long-label-that-should-not-exist-because-it-goes-against-the-spec.example.org.").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
|
|
#[test]
|
|
fn invalid_ascii() {
|
|
let err = validate_record_str("name.this-is-not-ascii-ß.example.org.").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
|
|
#[test]
|
|
fn invalid_octet() {
|
|
let err =
|
|
validate_record_str("name.this-character:-is-not-allowed.example.org.").unwrap_err();
|
|
assert_miette_snapshot!(err);
|
|
}
|
|
}
|