Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixed the record creation #22

Merged
merged 2 commits into from
Jul 30, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
32 changes: 24 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -35,16 +35,32 @@ log:

# The keys and domains configuration
keys:
- key1:
- domain1
- domain2
- key2:
- domain3
- domain4
key1:
sub.example.fr:
mname: ns-acme.example.fr.
rname: postmaster.example.fr.
example.fr:
mname: ns-acme.example.fr.
rname: postmaster.example.fr.
key2:
another-example.fr:
mname: ns-acme.another-example.fr.
rname: postmaster.another-example.fr.
fake.another-example.fr:
mname: ns-acme.another-example.fr.
rname: postmaster.another-example.fr.
```

In the previous example, the `dnsr` server will handle the domain1, domain2, domain3 and domain4 domains.
The key1 will be used to handle the domain1 and domain2 domains and the key2 will be used to handle the domain3 and domain4 domains.
In the previous example, the `dnsr` server will handle the `sub.example.fr`, `example.fr`, `another-example.fr` and `fake.another-example.fr` domains.
The key1 will be used to handle the `sub.example.fr` and `example.fr` domains and the key2 will be used to handle the `another-example.fr` and `fake.another-example.fr` domains.

The record created is the following for the `sub.example.fr` domain:

```text
_acme-challenge.sub.example.fr. 3600 IN SOA ns-acme.example.fr. postmaster.example.fr. 1722353587 10800 3600 605800 3600
```

**Note**: The prefix `_acme-challenge` is automatically added to the domain name.

**Note**: The dnsr server constantly whatches the `config.yml` file for changes.
If the file is modified, the server will reload the domains (e.g. add or remove domains).
Expand Down
15 changes: 12 additions & 3 deletions config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,16 @@ log:

keys:
key1:
- thibault-cne.fr
sub.example.fr:
mname: ns-acme.example.fr.
rname: postmaster.example.fr.
example.fr:
mname: ns-acme.example.fr.
rname: postmaster.example.fr.
key2:
- example.com
- example.org
another-example.fr:
mname: ns-acme.another-example.fr.
rname: postmaster.another-example.fr.
fake.another-example.fr:
mname: ns-acme.another-example.fr.
rname: postmaster.another-example.fr.
29 changes: 29 additions & 0 deletions src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ pub enum ErrorKind {
TSIGFileAlreadyExist,
RingUnspecified,
Base16,
Utf8,
}

impl std::fmt::Display for Error {
Expand All @@ -40,6 +41,7 @@ impl std::fmt::Display for ErrorKind {
TSIGFileAlreadyExist => write!(f, "tsig file already exists"),
RingUnspecified => write!(f, "ring unspecified error"),
Base16 => write!(f, "base16 error"),
Utf8 => write!(f, "utf8 error"),
}
}
}
Expand All @@ -53,6 +55,15 @@ impl From<ErrorKind> for Error {
}
}

impl From<std::str::Utf8Error> for Error {
fn from(value: std::str::Utf8Error) -> Self {
Self {
kind: ErrorKind::Utf8,
message: Some(value.to_string()),
}
}
}

impl From<notify::Error> for Error {
fn from(value: notify::Error) -> Self {
Self {
Expand Down Expand Up @@ -80,6 +91,15 @@ impl From<domain::base::name::FromStrError> for Error {
}
}

impl From<domain::base::name::NameError> for Error {
fn from(value: domain::base::name::NameError) -> Self {
Self {
kind: ErrorKind::DomainStr,
message: Some(value.to_string()),
}
}
}

impl From<domain::zonetree::error::ZoneTreeModificationError> for Error {
fn from(value: domain::zonetree::error::ZoneTreeModificationError) -> Self {
Self {
Expand All @@ -89,6 +109,15 @@ impl From<domain::zonetree::error::ZoneTreeModificationError> for Error {
}
}

impl From<domain::zonetree::error::OutOfZone> for Error {
fn from(_: domain::zonetree::error::OutOfZone) -> Self {
Self {
kind: ErrorKind::DomainZone,
message: Some("out of zone".to_string()),
}
}
}

impl From<std::io::Error> for Error {
fn from(value: std::io::Error) -> Self {
Self {
Expand Down
93 changes: 75 additions & 18 deletions src/key.rs
Original file line number Diff line number Diff line change
@@ -1,49 +1,106 @@
use core::str;
use std::collections::HashMap;
use std::ops::Deref;
use std::path::PathBuf;

use bytes::BytesMut;
use domain::base::iana::Class;
use domain::zonetree::types::StoredName;
use domain::zonetree::{Zone, ZoneBuilder};
use domain::base::{Record, Serial, Ttl};
use domain::rdata::Soa;
use domain::zonetree::types::{StoredName, StoredRecord};
use domain::zonetree::{Rrset, SharedRrset, Zone, ZoneBuilder};
use serde::Deserialize;

use crate::error::Result;

#[derive(Debug, Clone, Deserialize, Default)]
pub struct Keys(HashMap<KeyFile, Vec<Domain>>);
pub struct Keys(HashMap<KeyFile, HashMap<DomainName, DomainInfo>>);

impl Deref for Keys {
type Target = HashMap<KeyFile, Vec<Domain>>;
type Target = HashMap<KeyFile, HashMap<DomainName, DomainInfo>>;

fn deref(&self) -> &Self::Target {
&self.0
}
}

#[derive(Debug, Clone, Deserialize, PartialEq, Eq)]
#[serde(untagged)]
pub enum Domain {
Unamed(String),
pub struct DomainInfo {
mname: String,
rname: String,
}

pub trait TryIntoZones {
fn try_into_zones(self) -> Result<Vec<domain::zonetree::Zone>>;
#[derive(Debug, Clone, Deserialize, PartialEq, Eq, Hash)]
pub struct DomainName(String);

pub trait TryInto<T> {
fn try_into_t(self) -> Result<T>;
}

impl TryIntoZones for &[Domain] {
fn try_into_zones(self) -> Result<Vec<domain::zonetree::Zone>> {
self.iter().map(|d| d.try_into()).collect()
impl TryInto<Vec<domain::zonetree::Zone>> for &HashMap<DomainName, DomainInfo> {
fn try_into_t(self) -> Result<Vec<domain::zonetree::Zone>> {
self.iter().map(|d| d.try_into_t()).collect()
}
}

impl TryFrom<&Domain> for Zone {
impl TryFrom<&DomainInfo> for SharedRrset {
type Error = crate::error::Error;

fn try_from(value: &Domain) -> Result<Self> {
let apex_name = match value {
Domain::Unamed(name) => StoredName::bytes_from_str(name)?,
};
Ok(ZoneBuilder::new(apex_name, Class::IN).build())
fn try_from(value: &DomainInfo) -> std::result::Result<Self, Self::Error> {
let mut owner = BytesMut::with_capacity(16 + value.mname.len());
owner.extend_from_slice(b"_acme-challenge.");
owner.extend_from_slice(value.mname.as_bytes());

let record: StoredRecord = Record::new(
owner.freeze().try_into_t()?,
Class::IN,
Ttl::HOUR,
Soa::new(
(&value.mname).try_into_t()?,
(&value.rname).try_into_t()?,
Serial::now(),
Ttl::from_secs(10800),
Ttl::HOUR,
Ttl::from_secs(605800),
Ttl::HOUR,
)
.into(),
);
log::debug!(target: "record", "new record created: {:?}", record);
let rset: Rrset = record.into();

Ok(rset.into_shared())
}
}

impl TryInto<Zone> for (&DomainName, &DomainInfo) {
fn try_into_t(self) -> Result<Zone> {
let (name, info) = self;
let mut builder = ZoneBuilder::new(name.try_into_t()?, Class::IN);
builder.insert_rrset(&name.try_into_t()?, info.try_into()?)?;
let zone = builder.build();
log::debug!(target: "zone", "new zone created: {:?}", zone);
Ok(zone)
}
}

impl TryInto<StoredName> for &DomainName {
fn try_into_t(self) -> Result<StoredName> {
let mut owner = BytesMut::with_capacity(16 + self.0.len());
owner.extend_from_slice(b"_acme-challenge.");
owner.extend_from_slice(self.0.as_bytes());

owner.freeze().try_into_t()
}
}

impl<B> TryInto<StoredName> for B
where
B: AsRef<[u8]>,
{
fn try_into_t(self) -> Result<StoredName> {
let str = str::from_utf8(self.as_ref())?;
Ok(StoredName::bytes_from_str(str)?)
}
}

Expand Down
28 changes: 17 additions & 11 deletions src/watcher.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
use std::collections::HashMap;
use std::fs::File;
use std::path::Path;
use std::sync::mpsc::channel;
Expand All @@ -8,7 +9,7 @@ use notify::{Config, RecommendedWatcher, RecursiveMode, Watcher as NotifyWatcher

use crate::dns::State;
use crate::error::{ErrorKind, Result};
use crate::key::{Domain, KeyFile, Keys, TryIntoZones};
use crate::key::{DomainInfo, DomainName, KeyFile, Keys, TryInto};

#[derive(Debug, Clone)]
pub struct Watcher;
Expand Down Expand Up @@ -45,7 +46,7 @@ fn initialize_dns_zones(keys: &Keys, state: &Arc<State>) -> Result<()> {
}

for (k, v) in keys.iter() {
v.try_into_zones()?
v.try_into_t()?
.into_iter()
.try_for_each(|z| state.insert_zone(z))?;
match k.generate_key_file() {
Expand Down Expand Up @@ -85,10 +86,10 @@ fn handle_file_change(keys: &Keys, config_path: &Path, state: &Arc<State>) -> Re

fn handle_deleted_keys<'i, I>(state: &Arc<State>, deleted_keys: I) -> Result<()>
where
I: IntoIterator<Item = (&'i KeyFile, &'i Vec<Domain>)>,
I: IntoIterator<Item = (&'i KeyFile, &'i HashMap<DomainName, DomainInfo>)>,
{
for (k, v) in deleted_keys {
v.try_into_zones()?.into_iter().for_each(|z| {
v.try_into_t()?.into_iter().for_each(|z| {
let _ = state.remove_zone(z.apex_name(), z.class());
});

Expand All @@ -101,10 +102,10 @@ where

fn handle_added_keys<'i, I>(state: &Arc<State>, added_keys: I) -> Result<()>
where
I: IntoIterator<Item = (&'i KeyFile, &'i Vec<Domain>)>,
I: IntoIterator<Item = (&'i KeyFile, &'i HashMap<DomainName, DomainInfo>)>,
{
for (k, v) in added_keys {
v.try_into_zones()?.into_iter().for_each(|z| {
v.try_into_t()?.into_iter().for_each(|z| {
let _ = state.insert_zone(z);
});

Expand All @@ -117,20 +118,25 @@ where

fn handle_modified_keys<'i, I>(state: &Arc<State>, modified_keys: I) -> Result<()>
where
I: IntoIterator<Item = (&'i Vec<Domain>, &'i Vec<Domain>)>,
I: IntoIterator<
Item = (
&'i HashMap<DomainName, DomainInfo>,
&'i HashMap<DomainName, DomainInfo>,
),
>,
{
for (nv, ov) in modified_keys {
ov.iter()
.filter(|d| !nv.contains(d))
.filter(|&(d, _)| nv.get(d).is_none())
.try_for_each(|d| -> Result<()> {
let zone: Zone = d.try_into()?;
let zone: Zone = d.try_into_t()?;
let _ = state.remove_zone(zone.apex_name(), zone.class());
Ok(())
})?;
nv.iter()
.filter(|d| !ov.contains(d))
.filter(|&(d, _)| ov.get(d).is_none())
.try_for_each(|d| -> Result<()> {
let zone: Zone = d.try_into()?;
let zone: Zone = d.try_into_t()?;
let _ = state.insert_zone(zone);
Ok(())
})?;
Expand Down