use super::parser;
use crate::applayer::*;
use crate::conf::conf_get;
use crate::core::{ALPROTO_UNKNOWN, IPPROTO_TCP};
use crate::flow::Flow;
use nom7 as nom;
use std;
use std::collections::VecDeque;
use std::ffi::CString;
use std::os::raw::{c_char, c_int, c_void};
use suricata_sys::sys::{
AppLayerParserState, AppProto, SCAppLayerParserConfParserEnabled,
SCAppLayerParserRegisterLogger, SCAppLayerParserStateIssetFlag,
SCAppLayerProtoDetectConfProtoDetectionEnabled,
};
static mut TEMPLATE_MAX_TX: usize = 256;
pub(super) static mut ALPROTO_TEMPLATE: AppProto = ALPROTO_UNKNOWN;
#[derive(AppLayerEvent)]
enum TemplateEvent {
TooManyTransactions,
}
pub struct TemplateTransaction {
tx_id: u64,
pub request: Option<String>,
pub response: Option<String>,
tx_data: AppLayerTxData,
}
impl Default for TemplateTransaction {
fn default() -> Self {
Self::new()
}
}
impl TemplateTransaction {
pub fn new() -> TemplateTransaction {
Self {
tx_id: 0,
request: None,
response: None,
tx_data: AppLayerTxData::new(),
}
}
}
impl Transaction for TemplateTransaction {
fn id(&self) -> u64 {
self.tx_id
}
}
#[derive(Default)]
pub struct TemplateState {
state_data: AppLayerStateData,
tx_id: u64,
transactions: VecDeque<TemplateTransaction>,
request_gap: bool,
response_gap: bool,
}
impl State<TemplateTransaction> for TemplateState {
fn get_transaction_count(&self) -> usize {
self.transactions.len()
}
fn get_transaction_by_index(&self, index: usize) -> Option<&TemplateTransaction> {
self.transactions.get(index)
}
}
impl TemplateState {
pub fn new() -> Self {
Default::default()
}
fn free_tx(&mut self, tx_id: u64) {
let len = self.transactions.len();
let mut found = false;
let mut index = 0;
for i in 0..len {
let tx = &self.transactions[i];
if tx.tx_id == tx_id + 1 {
found = true;
index = i;
break;
}
}
if found {
self.transactions.remove(index);
}
}
pub fn get_tx(&mut self, tx_id: u64) -> Option<&TemplateTransaction> {
self.transactions.iter().find(|tx| tx.tx_id == tx_id + 1)
}
fn new_tx(&mut self) -> TemplateTransaction {
let mut tx = TemplateTransaction::new();
self.tx_id += 1;
tx.tx_id = self.tx_id;
return tx;
}
fn find_request(&mut self) -> Option<&mut TemplateTransaction> {
self.transactions
.iter_mut()
.find(|tx| tx.response.is_none())
}
fn parse_request(&mut self, input: &[u8]) -> AppLayerResult {
if input.is_empty() {
return AppLayerResult::ok();
}
if self.request_gap {
if probe(input).is_err() {
return AppLayerResult::ok();
}
self.request_gap = false;
}
let mut start = input;
while !start.is_empty() {
match parser::parse_message(start) {
Ok((rem, request)) => {
start = rem;
SCLogNotice!("Request: {}", request);
let mut tx = self.new_tx();
tx.request = Some(request);
if self.transactions.len() >= unsafe { TEMPLATE_MAX_TX } {
tx.tx_data
.set_event(TemplateEvent::TooManyTransactions as u8);
}
self.transactions.push_back(tx);
if self.transactions.len() >= unsafe { TEMPLATE_MAX_TX } {
return AppLayerResult::err();
}
}
Err(nom::Err::Incomplete(_)) => {
let consumed = input.len() - start.len();
let needed = start.len() + 1;
return AppLayerResult::incomplete(consumed as u32, needed as u32);
}
Err(_) => {
return AppLayerResult::err();
}
}
}
return AppLayerResult::ok();
}
fn parse_response(&mut self, input: &[u8]) -> AppLayerResult {
if input.is_empty() {
return AppLayerResult::ok();
}
if self.response_gap {
if probe(input).is_err() {
return AppLayerResult::ok();
}
self.response_gap = false;
}
let mut start = input;
while !start.is_empty() {
match parser::parse_message(start) {
Ok((rem, response)) => {
start = rem;
if let Some(tx) = self.find_request() {
tx.tx_data.updated_tc = true;
tx.response = Some(response);
SCLogNotice!("Found response for request:");
SCLogNotice!("- Request: {:?}", tx.request);
SCLogNotice!("- Response: {:?}", tx.response);
}
}
Err(nom::Err::Incomplete(_)) => {
let consumed = input.len() - start.len();
let needed = start.len() + 1;
return AppLayerResult::incomplete(consumed as u32, needed as u32);
}
Err(_) => {
return AppLayerResult::err();
}
}
}
return AppLayerResult::ok();
}
fn on_request_gap(&mut self, _size: u32) {
self.request_gap = true;
}
fn on_response_gap(&mut self, _size: u32) {
self.response_gap = true;
}
}
fn probe(input: &[u8]) -> nom::IResult<&[u8], ()> {
let size = std::cmp::min(10, input.len());
let (rem, prefix) = nom::bytes::complete::take(size)(input)?;
nom::sequence::terminated(
nom::bytes::complete::take_while1(nom::character::is_digit),
nom::bytes::complete::tag(":"),
)(prefix)?;
Ok((rem, ()))
}
unsafe extern "C" fn template_probing_parser(
_flow: *const Flow, _direction: u8, input: *const u8, input_len: u32, _rdir: *mut u8,
) -> AppProto {
if input_len > 1 && !input.is_null() {
let slice = build_slice!(input, input_len as usize);
if probe(slice).is_ok() {
return ALPROTO_TEMPLATE;
}
}
return ALPROTO_UNKNOWN;
}
extern "C" fn template_state_new(_orig_state: *mut c_void, _orig_proto: AppProto) -> *mut c_void {
let state = TemplateState::new();
let boxed = Box::new(state);
return Box::into_raw(boxed) as *mut c_void;
}
unsafe extern "C" fn template_state_free(state: *mut c_void) {
std::mem::drop(Box::from_raw(state as *mut TemplateState));
}
unsafe extern "C" fn template_state_tx_free(state: *mut c_void, tx_id: u64) {
let state = cast_pointer!(state, TemplateState);
state.free_tx(tx_id);
}
unsafe extern "C" fn template_parse_request(
_flow: *mut Flow, state: *mut c_void, pstate: *mut AppLayerParserState,
stream_slice: StreamSlice, _data: *const c_void,
) -> AppLayerResult {
let eof = SCAppLayerParserStateIssetFlag(pstate, APP_LAYER_PARSER_EOF_TS) > 0;
if eof {
return AppLayerResult::ok();
}
let state = cast_pointer!(state, TemplateState);
if stream_slice.is_gap() {
state.on_request_gap(stream_slice.gap_size());
AppLayerResult::ok()
} else {
let buf = stream_slice.as_slice();
state.parse_request(buf)
}
}
unsafe extern "C" fn template_parse_response(
_flow: *mut Flow, state: *mut c_void, pstate: *mut AppLayerParserState,
stream_slice: StreamSlice, _data: *const c_void,
) -> AppLayerResult {
let _eof = SCAppLayerParserStateIssetFlag(pstate, APP_LAYER_PARSER_EOF_TC) > 0;
let state = cast_pointer!(state, TemplateState);
if stream_slice.is_gap() {
state.on_response_gap(stream_slice.gap_size());
AppLayerResult::ok()
} else {
let buf = stream_slice.as_slice();
state.parse_response(buf)
}
}
unsafe extern "C" fn template_state_get_tx(state: *mut c_void, tx_id: u64) -> *mut c_void {
let state = cast_pointer!(state, TemplateState);
match state.get_tx(tx_id) {
Some(tx) => {
return tx as *const _ as *mut _;
}
None => {
return std::ptr::null_mut();
}
}
}
unsafe extern "C" fn template_state_get_tx_count(state: *mut c_void) -> u64 {
let state = cast_pointer!(state, TemplateState);
return state.tx_id;
}
unsafe extern "C" fn template_tx_get_alstate_progress(tx: *mut c_void, _direction: u8) -> c_int {
let tx = cast_pointer!(tx, TemplateTransaction);
if tx.response.is_some() {
return 1;
}
return 0;
}
export_tx_data_get!(template_get_tx_data, TemplateTransaction);
export_state_data_get!(template_get_state_data, TemplateState);
const PARSER_NAME: &[u8] = b"template\0";
#[no_mangle]
pub unsafe extern "C" fn SCRegisterTemplateParser() {
if crate::conf::conf_get_node("app-layer.protocols.template").is_none() {
return;
}
let default_port = CString::new("[7000]").unwrap();
let parser = RustParser {
name: PARSER_NAME.as_ptr() as *const c_char,
default_port: default_port.as_ptr(),
ipproto: IPPROTO_TCP,
probe_ts: Some(template_probing_parser),
probe_tc: Some(template_probing_parser),
min_depth: 0,
max_depth: 16,
state_new: template_state_new,
state_free: template_state_free,
tx_free: template_state_tx_free,
parse_ts: template_parse_request,
parse_tc: template_parse_response,
get_tx_count: template_state_get_tx_count,
get_tx: template_state_get_tx,
tx_comp_st_ts: 1,
tx_comp_st_tc: 1,
tx_get_progress: template_tx_get_alstate_progress,
get_eventinfo: Some(TemplateEvent::get_event_info),
get_eventinfo_byid: Some(TemplateEvent::get_event_info_by_id),
localstorage_new: None,
localstorage_free: None,
get_tx_files: None,
get_tx_iterator: Some(state_get_tx_iterator::<TemplateState, TemplateTransaction>),
get_tx_data: template_get_tx_data,
get_state_data: template_get_state_data,
apply_tx_config: None,
flags: APP_LAYER_PARSER_OPT_ACCEPT_GAPS,
get_frame_id_by_name: None,
get_frame_name_by_id: None,
get_state_id_by_name: None,
get_state_name_by_id: None,
};
let ip_proto_str = CString::new("tcp").unwrap();
if SCAppLayerProtoDetectConfProtoDetectionEnabled(ip_proto_str.as_ptr(), parser.name) != 0 {
let alproto = AppLayerRegisterProtocolDetection(&parser, 1);
ALPROTO_TEMPLATE = alproto;
if SCAppLayerParserConfParserEnabled(ip_proto_str.as_ptr(), parser.name) != 0 {
let _ = AppLayerRegisterParser(&parser, alproto);
}
if let Some(val) = conf_get("app-layer.protocols.template.max-tx") {
if let Ok(v) = val.parse::<usize>() {
TEMPLATE_MAX_TX = v;
} else {
SCLogError!("Invalid value for template.max-tx");
}
}
SCAppLayerParserRegisterLogger(IPPROTO_TCP, ALPROTO_TEMPLATE);
SCLogNotice!("Rust template parser registered.");
} else {
SCLogNotice!("Protocol detector and parser disabled for TEMPLATE.");
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_probe() {
assert!(probe(b"1").is_err());
assert!(probe(b"1:").is_ok());
assert!(probe(b"123456789:").is_ok());
assert!(probe(b"0123456789:").is_err());
}
#[test]
fn test_incomplete() {
let mut state = TemplateState::new();
let buf = b"5:Hello3:bye";
let r = state.parse_request(&buf[0..0]);
assert_eq!(
r,
AppLayerResult {
status: 0,
consumed: 0,
needed: 0
}
);
let r = state.parse_request(&buf[0..1]);
assert_eq!(
r,
AppLayerResult {
status: 1,
consumed: 0,
needed: 2
}
);
let r = state.parse_request(&buf[0..2]);
assert_eq!(
r,
AppLayerResult {
status: 1,
consumed: 0,
needed: 3
}
);
let r = state.parse_request(&buf[0..7]);
assert_eq!(
r,
AppLayerResult {
status: 0,
consumed: 0,
needed: 0
}
);
let r = state.parse_request(&buf[0..9]);
assert_eq!(
r,
AppLayerResult {
status: 1,
consumed: 7,
needed: 3
}
);
}
}