|
|
|
@ -18,10 +18,10 @@
|
|
|
|
|
use std;
|
|
|
|
|
use std::os::raw::c_void;
|
|
|
|
|
|
|
|
|
|
use crate::conf::ConfNode;
|
|
|
|
|
use crate::dhcp::dhcp::*;
|
|
|
|
|
use crate::dhcp::parser::{DHCPOptionWrapper,DHCPOptGeneric};
|
|
|
|
|
use crate::dhcp::parser::{DHCPOptGeneric, DHCPOptionWrapper};
|
|
|
|
|
use crate::dns::log::dns_print_addr;
|
|
|
|
|
use crate::conf::ConfNode;
|
|
|
|
|
use crate::jsonbuilder::{JsonBuilder, JsonError};
|
|
|
|
|
|
|
|
|
|
pub struct DHCPLogger {
|
|
|
|
@ -29,11 +29,10 @@ pub struct DHCPLogger {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl DHCPLogger {
|
|
|
|
|
|
|
|
|
|
pub fn new(conf: ConfNode) -> Self {
|
|
|
|
|
return Self {
|
|
|
|
|
extended: conf.get_child_bool("extended"),
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn get_type(&self, tx: &DHCPTransaction) -> Option<u8> {
|
|
|
|
@ -42,7 +41,8 @@ impl DHCPLogger {
|
|
|
|
|
let code = option.code;
|
|
|
|
|
#[allow(clippy::single_match)]
|
|
|
|
|
match &option.option {
|
|
|
|
|
DHCPOptionWrapper::Generic(option) => {
|
|
|
|
|
DHCPOptionWrapper::Generic(option) =>
|
|
|
|
|
{
|
|
|
|
|
#[allow(clippy::single_match)]
|
|
|
|
|
match code {
|
|
|
|
|
DHCP_OPT_TYPE => {
|
|
|
|
@ -61,7 +61,7 @@ impl DHCPLogger {
|
|
|
|
|
|
|
|
|
|
pub fn do_log(&self, tx: &DHCPTransaction) -> bool {
|
|
|
|
|
if !self.extended {
|
|
|
|
|
if let Some(DHCP_TYPE_ACK) = self.get_type(tx){
|
|
|
|
|
if let Some(DHCP_TYPE_ACK) = self.get_type(tx) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
@ -86,101 +86,86 @@ impl DHCPLogger {
|
|
|
|
|
js.set_string("type", "<unknown>")?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
js.set_uint("id", header.txid as u64)?;
|
|
|
|
|
js.set_string("client_mac",
|
|
|
|
|
&format_addr_hex(&header.clienthw))?;
|
|
|
|
|
js.set_string("client_mac", &format_addr_hex(&header.clienthw))?;
|
|
|
|
|
js.set_string("assigned_ip", &dns_print_addr(&header.yourip))?;
|
|
|
|
|
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_string("client_ip", &dns_print_addr(&header.clientip))?;
|
|
|
|
|
if header.opcode == BOOTP_REPLY {
|
|
|
|
|
js.set_string("relay_ip",
|
|
|
|
|
&dns_print_addr(&header.giaddr))?;
|
|
|
|
|
js.set_string("next_server_ip",
|
|
|
|
|
&dns_print_addr(&header.serverip))?;
|
|
|
|
|
js.set_string("relay_ip", &dns_print_addr(&header.giaddr))?;
|
|
|
|
|
js.set_string("next_server_ip", &dns_print_addr(&header.serverip))?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for option in options {
|
|
|
|
|
let code = option.code;
|
|
|
|
|
match option.option {
|
|
|
|
|
DHCPOptionWrapper::ClientId(ref clientid) => {
|
|
|
|
|
js.set_string("client_id",
|
|
|
|
|
&format_addr_hex(&clientid.data))?;
|
|
|
|
|
js.set_string("client_id", &format_addr_hex(&clientid.data))?;
|
|
|
|
|
}
|
|
|
|
|
DHCPOptionWrapper::TimeValue(ref time_value) => {
|
|
|
|
|
match code {
|
|
|
|
|
DHCP_OPT_ADDRESS_TIME => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_uint("lease_time",
|
|
|
|
|
time_value.seconds as u64)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_REBINDING_TIME => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_uint("rebinding_time",
|
|
|
|
|
time_value.seconds as u64)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_RENEWAL_TIME => {
|
|
|
|
|
js.set_uint("renewal_time",
|
|
|
|
|
time_value.seconds as u64)?;
|
|
|
|
|
DHCPOptionWrapper::TimeValue(ref time_value) => match code {
|
|
|
|
|
DHCP_OPT_ADDRESS_TIME => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_uint("lease_time", time_value.seconds as u64)?;
|
|
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCPOptionWrapper::Generic(ref option) => {
|
|
|
|
|
match code {
|
|
|
|
|
DHCP_OPT_SUBNET_MASK => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_string("subnet_mask",
|
|
|
|
|
&dns_print_addr(&option.data))?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_REBINDING_TIME => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_uint("rebinding_time", time_value.seconds as u64)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_HOSTNAME => {
|
|
|
|
|
if !option.data.is_empty() {
|
|
|
|
|
js.set_string_from_bytes("hostname",
|
|
|
|
|
&option.data)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_RENEWAL_TIME => {
|
|
|
|
|
js.set_uint("renewal_time", time_value.seconds as u64)?;
|
|
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
},
|
|
|
|
|
DHCPOptionWrapper::Generic(ref option) => match code {
|
|
|
|
|
DHCP_OPT_SUBNET_MASK => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_string("subnet_mask", &dns_print_addr(&option.data))?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_TYPE => {
|
|
|
|
|
self.log_opt_type(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_HOSTNAME => {
|
|
|
|
|
if !option.data.is_empty() {
|
|
|
|
|
js.set_string_from_bytes("hostname", &option.data)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_REQUESTED_IP => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_string("requested_ip",
|
|
|
|
|
&dns_print_addr(&option.data))?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_TYPE => {
|
|
|
|
|
self.log_opt_type(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_REQUESTED_IP => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
js.set_string("requested_ip", &dns_print_addr(&option.data))?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_PARAMETER_LIST => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_parameters(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_PARAMETER_LIST => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_parameters(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_DNS_SERVER => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_dns_server(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_DNS_SERVER => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_dns_server(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_ROUTERS => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_routers(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_ROUTERS => {
|
|
|
|
|
if self.extended {
|
|
|
|
|
self.log_opt_routers(js, option)?;
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_VENDOR_CLASS_ID => {
|
|
|
|
|
if self.extended && !option.data.is_empty(){
|
|
|
|
|
js.set_string_from_bytes("vendor_class_identifier",
|
|
|
|
|
&option.data)?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
DHCP_OPT_VENDOR_CLASS_ID => {
|
|
|
|
|
if self.extended && !option.data.is_empty() {
|
|
|
|
|
js.set_string_from_bytes("vendor_class_identifier", &option.data)?;
|
|
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_ => {}
|
|
|
|
|
},
|
|
|
|
|
_ => {}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
js.close()?;
|
|
|
|
|
|
|
|
|
|
return Ok(());
|
|
|
|
@ -197,14 +182,16 @@ impl DHCPLogger {
|
|
|
|
|
DHCP_TYPE_NAK => "nak",
|
|
|
|
|
DHCP_TYPE_RELEASE => "release",
|
|
|
|
|
DHCP_TYPE_INFORM => "inform",
|
|
|
|
|
_ => "unknown"
|
|
|
|
|
_ => "unknown",
|
|
|
|
|
};
|
|
|
|
|
js.set_string("dhcp_type", dhcp_type)?;
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn log_opt_parameters(&self, js: &mut JsonBuilder, option: &DHCPOptGeneric) -> Result<(), JsonError> {
|
|
|
|
|
fn log_opt_parameters(
|
|
|
|
|
&self, js: &mut JsonBuilder, option: &DHCPOptGeneric,
|
|
|
|
|
) -> Result<(), JsonError> {
|
|
|
|
|
js.open_array("params")?;
|
|
|
|
|
for i in &option.data {
|
|
|
|
|
let param = match *i {
|
|
|
|
@ -216,7 +203,7 @@ impl DHCPLogger {
|
|
|
|
|
DHCP_PARAM_NTP_SERVER => "ntp_server",
|
|
|
|
|
DHCP_PARAM_TFTP_SERVER_NAME => "tftp_server_name",
|
|
|
|
|
DHCP_PARAM_TFTP_SERVER_IP => "tftp_server_ip",
|
|
|
|
|
_ => ""
|
|
|
|
|
_ => "",
|
|
|
|
|
};
|
|
|
|
|
if !param.is_empty() {
|
|
|
|
|
js.append_string(param)?;
|
|
|
|
@ -225,8 +212,10 @@ impl DHCPLogger {
|
|
|
|
|
js.close()?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn log_opt_dns_server(&self, js: &mut JsonBuilder, option: &DHCPOptGeneric) -> Result<(), JsonError> {
|
|
|
|
|
|
|
|
|
|
fn log_opt_dns_server(
|
|
|
|
|
&self, js: &mut JsonBuilder, option: &DHCPOptGeneric,
|
|
|
|
|
) -> Result<(), JsonError> {
|
|
|
|
|
js.open_array("dns_servers")?;
|
|
|
|
|
for i in 0..(option.data.len() / 4) {
|
|
|
|
|
let val = dns_print_addr(&option.data[(i * 4)..(i * 4) + 4]);
|
|
|
|
@ -235,8 +224,10 @@ impl DHCPLogger {
|
|
|
|
|
js.close()?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn log_opt_routers(&self, js: &mut JsonBuilder, option: &DHCPOptGeneric) -> Result<(), JsonError> {
|
|
|
|
|
|
|
|
|
|
fn log_opt_routers(
|
|
|
|
|
&self, js: &mut JsonBuilder, option: &DHCPOptGeneric,
|
|
|
|
|
) -> Result<(), JsonError> {
|
|
|
|
|
js.open_array("routers")?;
|
|
|
|
|
for i in 0..(option.data.len() / 4) {
|
|
|
|
|
let val = dns_print_addr(&option.data[(i * 4)..(i * 4) + 4]);
|
|
|
|
@ -245,13 +236,10 @@ impl DHCPLogger {
|
|
|
|
|
js.close()?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn format_addr_hex(input: &[u8]) -> String {
|
|
|
|
|
let parts: Vec<String> = input.iter()
|
|
|
|
|
.map(|b| format!("{:02x}", b))
|
|
|
|
|
.collect();
|
|
|
|
|
let parts: Vec<String> = input.iter().map(|b| format!("{:02x}", b)).collect();
|
|
|
|
|
return parts.join(":");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -268,18 +256,18 @@ pub unsafe extern "C" fn SCDhcpLoggerFree(logger: *mut std::os::raw::c_void) {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[no_mangle]
|
|
|
|
|
pub unsafe extern "C" fn SCDhcpLoggerLog(logger: *mut std::os::raw::c_void,
|
|
|
|
|
tx: *mut std::os::raw::c_void,
|
|
|
|
|
js: &mut JsonBuilder) -> bool {
|
|
|
|
|
pub unsafe extern "C" fn SCDhcpLoggerLog(
|
|
|
|
|
logger: *mut std::os::raw::c_void, tx: *mut std::os::raw::c_void, js: &mut JsonBuilder,
|
|
|
|
|
) -> bool {
|
|
|
|
|
let logger = cast_pointer!(logger, DHCPLogger);
|
|
|
|
|
let tx = cast_pointer!(tx, DHCPTransaction);
|
|
|
|
|
logger.log(tx, js).is_ok()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[no_mangle]
|
|
|
|
|
pub unsafe extern "C" fn SCDhcpLoggerDoLog(logger: *mut std::os::raw::c_void,
|
|
|
|
|
tx: *mut std::os::raw::c_void)
|
|
|
|
|
-> bool {
|
|
|
|
|
pub unsafe extern "C" fn SCDhcpLoggerDoLog(
|
|
|
|
|
logger: *mut std::os::raw::c_void, tx: *mut std::os::raw::c_void,
|
|
|
|
|
) -> bool {
|
|
|
|
|
let logger = cast_pointer!(logger, DHCPLogger);
|
|
|
|
|
let tx = cast_pointer!(tx, DHCPTransaction);
|
|
|
|
|
logger.do_log(tx)
|
|
|
|
|