mirror of
https://github.com/ZeusWPI/ZNS.git
synced 2025-01-05 06:19:44 +01:00
respond refused if query is not in authoritative zone
This commit is contained in:
parent
8e46045d4c
commit
4261d06248
10 changed files with 67 additions and 30 deletions
|
@ -177,7 +177,7 @@ mod tests {
|
|||
fn test() {
|
||||
let mut connection = get_test_connection();
|
||||
|
||||
let rr = get_rr();
|
||||
let rr = get_rr(None);
|
||||
|
||||
let f = |connection: &mut PgConnection| {
|
||||
get_from_database(
|
||||
|
|
|
@ -5,7 +5,7 @@ use zns::{
|
|||
structs::{Message, Question, RR},
|
||||
};
|
||||
|
||||
use crate::db::models::get_from_database;
|
||||
use crate::{config::Config, db::models::get_from_database};
|
||||
|
||||
use super::ResponseHandler;
|
||||
|
||||
|
@ -20,6 +20,8 @@ impl ResponseHandler for QueryHandler {
|
|||
) -> Result<Message, ZNSError> {
|
||||
let mut response = message.clone();
|
||||
|
||||
message.check_authoritative(&Config::get().authoritative_zone)?;
|
||||
|
||||
for question in &message.question {
|
||||
let answers = get_from_database(
|
||||
&question.qname,
|
||||
|
@ -90,8 +92,8 @@ mod tests {
|
|||
#[tokio::test]
|
||||
async fn test_handle_query() {
|
||||
let mut connection = get_test_connection();
|
||||
let rr = get_rr();
|
||||
let mut message = get_message();
|
||||
let rr = get_rr(Some(Config::get().authoritative_zone.clone()));
|
||||
let mut message = get_message(Some(Config::get().authoritative_zone.clone()));
|
||||
message.header.ancount = 0;
|
||||
message.answer = vec![];
|
||||
|
||||
|
|
|
@ -3,11 +3,10 @@ use diesel::PgConnection;
|
|||
use crate::{
|
||||
config::Config,
|
||||
db::models::{delete_from_database, insert_into_database},
|
||||
utils::vec_equal,
|
||||
};
|
||||
|
||||
use zns::errors::ZNSError;
|
||||
use zns::structs::{Class, Message, RRClass, RRType, Type};
|
||||
use zns::{errors::ZNSError, utils::vec_equal};
|
||||
|
||||
use self::sig::Sig;
|
||||
|
||||
|
@ -37,18 +36,13 @@ impl ResponseHandler for UpdateHandler {
|
|||
}
|
||||
|
||||
// Check Zone authority
|
||||
let zone = &message.question[0];
|
||||
let zlen = zone.qname.len();
|
||||
let auth_zone = &Config::get().authoritative_zone;
|
||||
if !(zlen >= auth_zone.len() && vec_equal(&zone.qname[zlen - auth_zone.len()..], auth_zone))
|
||||
{
|
||||
return Err(ZNSError::Formerr {
|
||||
message: "Invalid zone".to_string(),
|
||||
});
|
||||
}
|
||||
message.check_authoritative(&Config::get().authoritative_zone)?;
|
||||
|
||||
// Check Prerequisite TODO: implement this
|
||||
|
||||
let zone = &message.question[0];
|
||||
let zlen = zone.qname.len();
|
||||
|
||||
//TODO: this code is ugly
|
||||
let last = message.additional.last();
|
||||
if last.is_some() && last.unwrap()._type == Type::Type(RRType::SIG) {
|
||||
|
|
|
@ -4,7 +4,6 @@ mod config;
|
|||
mod db;
|
||||
mod handlers;
|
||||
mod resolver;
|
||||
mod utils;
|
||||
|
||||
use config::Config;
|
||||
|
||||
|
|
|
@ -104,6 +104,8 @@ pub async fn tcp_listener_loop(addr: SocketAddr) -> Result<(), Box<dyn Error>> {
|
|||
mod tests {
|
||||
use zns::structs::{Class, Question, RRClass, RRType, Type};
|
||||
|
||||
use crate::config::Config;
|
||||
|
||||
use super::*;
|
||||
|
||||
#[tokio::test]
|
||||
|
@ -118,7 +120,7 @@ mod tests {
|
|||
arcount: 0,
|
||||
},
|
||||
question: vec![Question {
|
||||
qname: vec![String::from("example"), String::from("org")],
|
||||
qname: Config::get().authoritative_zone.clone(),
|
||||
qtype: Type::Type(RRType::A),
|
||||
qclass: Class::Class(RRClass::IN),
|
||||
}],
|
||||
|
|
|
@ -5,3 +5,4 @@ pub mod reader;
|
|||
pub mod structs;
|
||||
|
||||
pub mod test_utils;
|
||||
pub mod utils;
|
||||
|
|
|
@ -1,4 +1,8 @@
|
|||
use crate::structs::{Message, Opcode, RCODE};
|
||||
use crate::{
|
||||
errors::ZNSError,
|
||||
structs::{LabelString, Message, Opcode, RCODE},
|
||||
utils::vec_equal,
|
||||
};
|
||||
|
||||
impl Message {
|
||||
pub fn set_response(&mut self, rcode: RCODE) {
|
||||
|
@ -10,16 +14,32 @@ impl Message {
|
|||
Opcode::try_from((self.header.flags & 0b0111100000000000) >> 11)
|
||||
}
|
||||
|
||||
#[allow(dead_code)] // Used with tests
|
||||
#[cfg(feature = "test-utils")]
|
||||
pub fn get_rcode(&self) -> Result<RCODE, u16> {
|
||||
RCODE::try_from(self.header.flags & (!0 >> 12))
|
||||
}
|
||||
|
||||
pub fn check_authoritative(&self, auth_zone: &LabelString) -> Result<(), ZNSError> {
|
||||
let authoritative = self.question.iter().all(|question| {
|
||||
let zlen = question.qname.len();
|
||||
zlen >= auth_zone.len()
|
||||
&& vec_equal(&question.qname[zlen - auth_zone.len()..], auth_zone)
|
||||
});
|
||||
|
||||
if !authoritative {
|
||||
return Err(ZNSError::Refused {
|
||||
message: "Not authoritative".to_string(),
|
||||
});
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
|
||||
use crate::structs::Header;
|
||||
use crate::{structs::Header, test_utils::get_message};
|
||||
|
||||
use super::*;
|
||||
|
||||
|
@ -48,4 +68,19 @@ mod tests {
|
|||
|
||||
assert_eq!(message.get_rcode().unwrap(), RCODE::NOTIMP);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_not_authoritative() {
|
||||
let message = get_message(Some(vec![
|
||||
String::from("not"),
|
||||
String::from("good"),
|
||||
String::from("zone"),
|
||||
]));
|
||||
|
||||
let zone = vec![String::from("good")];
|
||||
|
||||
assert!(message
|
||||
.check_authoritative(&zone)
|
||||
.is_err_and(|x| x.rcode() == RCODE::REFUSED));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -328,7 +328,7 @@ pub mod tests {
|
|||
|
||||
#[test]
|
||||
fn test_parse_rr() {
|
||||
let rr = get_rr();
|
||||
let rr = get_rr(None);
|
||||
|
||||
let bytes = RR::to_bytes(rr.clone());
|
||||
let parsed = RR::from_bytes(&mut Reader::new(&bytes));
|
||||
|
@ -392,7 +392,7 @@ pub mod tests {
|
|||
|
||||
#[test]
|
||||
fn test_parse_message() {
|
||||
let message = get_message();
|
||||
let message = get_message(None);
|
||||
let bytes = Message::to_bytes(message.clone());
|
||||
let parsed = Message::from_bytes(&mut Reader::new(&bytes));
|
||||
assert!(parsed.is_ok());
|
||||
|
|
|
@ -2,9 +2,9 @@
|
|||
use crate::structs::*;
|
||||
|
||||
#[cfg(feature = "test-utils")]
|
||||
pub fn get_rr() -> RR {
|
||||
pub fn get_rr(name: Option<LabelString>) -> RR {
|
||||
RR {
|
||||
name: vec![String::from("example"), String::from("org")],
|
||||
name: name.unwrap_or(vec![String::from("example"), String::from("org")]),
|
||||
_type: Type::Type(RRType::A),
|
||||
class: Class::Class(RRClass::IN),
|
||||
ttl: 10,
|
||||
|
@ -13,7 +13,7 @@ pub fn get_rr() -> RR {
|
|||
}
|
||||
}
|
||||
|
||||
pub fn get_message() -> Message {
|
||||
pub fn get_message(name: Option<LabelString>) -> Message {
|
||||
Message {
|
||||
header: Header {
|
||||
id: 1,
|
||||
|
@ -25,18 +25,22 @@ pub fn get_message() -> Message {
|
|||
},
|
||||
question: vec![
|
||||
Question {
|
||||
qname: vec![String::from("example"), String::from("org")],
|
||||
qname: name
|
||||
.clone()
|
||||
.unwrap_or(vec![String::from("example"), String::from("org")]),
|
||||
qtype: Type::Type(RRType::A),
|
||||
qclass: Class::Class(RRClass::IN),
|
||||
},
|
||||
Question {
|
||||
qname: vec![String::from("example"), String::from("org")],
|
||||
qname: name
|
||||
.clone()
|
||||
.unwrap_or(vec![String::from("example"), String::from("org")]),
|
||||
qtype: Type::Type(RRType::A),
|
||||
qclass: Class::Class(RRClass::IN),
|
||||
},
|
||||
],
|
||||
answer: vec![get_rr()],
|
||||
authority: vec![get_rr()],
|
||||
additional: vec![get_rr()],
|
||||
answer: vec![get_rr(name.clone())],
|
||||
authority: vec![get_rr(name.clone())],
|
||||
additional: vec![get_rr(name)],
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue