use nom7::bytes::complete::{tag, take_until};
use nom7::character::complete::{digit1, multispace0};
use nom7::combinator::{complete, map_res, opt, verify};
use nom7::sequence::{delimited, tuple};
use nom7::{Err, IResult};
use std;
use std::str;
use std::str::FromStr;
pub mod constant;
pub mod event;
pub mod ftp;
pub mod response;
fn getu16(i: &[u8]) -> IResult<&[u8], u16> {
map_res(
map_res(delimited(multispace0, digit1, multispace0), str::from_utf8),
FromStr::from_str,
)(i)
}
fn parse_u16(i: &[u8]) -> IResult<&[u8], u16> {
map_res(map_res(digit1, str::from_utf8), u16::from_str)(i)
}
fn ftp_active_port(i: &[u8]) -> IResult<&[u8], u16> {
let (i, _) = tag("PORT")(i)?;
let (i, _) = delimited(multispace0, digit1, multispace0)(i)?;
let (i, _) = tuple((
tag(","),
digit1,
tag(","),
digit1,
tag(","),
digit1,
tag(","),
))(i)?;
let (i, part1) = verify(parse_u16, |&v| v <= u8::MAX as u16)(i)?;
let (i, _) = tag(",")(i)?;
let (i, part2) = verify(parse_u16, |&v| v <= u8::MAX as u16)(i)?;
Ok((i, part1 * 256 + part2))
}
fn ftp_pasv_response(i: &[u8]) -> IResult<&[u8], u16> {
let (i, _) = tag("227")(i)?;
let (i, _) = take_until("(")(i)?;
let (i, _) = tag("(")(i)?;
let (i, _) = tuple((
digit1,
tag(","),
digit1,
tag(","),
digit1,
tag(","),
digit1,
tag(","),
))(i)?;
let (i, part1) = verify(getu16, |&v| v <= u8::MAX as u16)(i)?;
let (i, _) = tag(",")(i)?;
let (i, part2) = verify(getu16, |&v| v <= u8::MAX as u16)(i)?;
let (i, _) = tag(")")(i)?;
let (i, _) = opt(complete(tag(".")))(i)?;
Ok((i, part1 * 256 + part2))
}
#[no_mangle]
pub unsafe extern "C" fn SCFTPParsePort(input: *const u8, len: u32) -> u16 {
if input.is_null() {
return 0;
}
let buf = build_slice!(input, len as usize);
match ftp_active_port(buf) {
Ok((_, dport)) => {
return dport;
}
Err(Err::Incomplete(_)) => {
SCLogDebug!("port incomplete: '{:?}'", buf);
}
Err(_) => {
SCLogDebug!("port error on '{:?}'", buf);
}
}
return 0;
}
#[no_mangle]
pub unsafe extern "C" fn SCFTPParsePortPasv(input: *const u8, len: u32) -> u16 {
if input.is_null() {
return 0;
}
let buf = build_slice!(input, len as usize);
match ftp_pasv_response(buf) {
Ok((_, dport)) => {
return dport;
}
Err(Err::Incomplete(_)) => {
SCLogDebug!("pasv incomplete: '{:?}'", String::from_utf8_lossy(buf));
}
Err(_) => {
SCLogDebug!("pasv error on '{:?}'", String::from_utf8_lossy(buf));
}
}
return 0;
}
pub fn ftp_epsv_response(i: &[u8]) -> IResult<&[u8], u16> {
let (i, _) = tag("229")(i)?;
let (i, _) = take_until("|||")(i)?;
let (i, _) = tag("|||")(i)?;
let (i, port) = getu16(i)?;
let (i, _) = tag("|)")(i)?;
let (i, _) = opt(complete(tag(".")))(i)?;
Ok((i, port))
}
fn ftp_active_eprt(i: &[u8]) -> IResult<&[u8], u16> {
let (i, _) = tag("EPRT")(i)?;
let (i, _) = take_until("|")(i)?;
let (i, _) = tag("|")(i)?;
let (i, _) = take_until("|")(i)?;
let (i, _) = tag("|")(i)?;
let (i, _) = take_until("|")(i)?;
let (i, _) = tag("|")(i)?;
let (i, port) = getu16(i)?;
let (i, _) = tag("|")(i)?;
Ok((i, port))
}
#[no_mangle]
pub unsafe extern "C" fn SCFTPParsePortEprt(input: *const u8, len: u32) -> u16 {
if input.is_null() {
return 0;
}
let buf = build_slice!(input, len as usize);
match ftp_active_eprt(buf) {
Ok((_, dport)) => {
return dport;
}
Err(Err::Incomplete(_)) => {
SCLogDebug!("eprt incomplete: '{:?}'", String::from_utf8_lossy(buf));
}
Err(_) => {
SCLogDebug!("epsv incomplete: '{:?}'", String::from_utf8_lossy(buf));
}
}
return 0;
}
#[no_mangle]
pub unsafe extern "C" fn SCFTPParsePortEpsv(input: *const u8, len: u32) -> u16 {
if input.is_null() {
return 0;
}
let buf = build_slice!(input, len as usize);
match ftp_epsv_response(buf) {
Ok((_, dport)) => {
return dport;
}
Err(Err::Incomplete(_)) => {
SCLogDebug!("epsv incomplete: '{:?}'", String::from_utf8_lossy(buf));
}
Err(_) => {
SCLogDebug!("epsv incomplete: '{:?}'", String::from_utf8_lossy(buf));
}
}
return 0;
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_pasv_response_valid() {
let port =
ftp_pasv_response("227 Entering Passive Mode (212,27,32,66,221,243).".as_bytes());
assert_eq!(port, Ok((&b""[..], 56819)));
let port_notdot =
ftp_pasv_response("227 Entering Passive Mode (212,27,32,66,221,243)".as_bytes());
assert_eq!(port_notdot, Ok((&b""[..], 56819)));
let port_epsv_dot =
ftp_epsv_response("229 Entering Extended Passive Mode (|||48758|).".as_bytes());
assert_eq!(port_epsv_dot, Ok((&b""[..], 48758)));
let port_epsv_nodot =
ftp_epsv_response("229 Entering Extended Passive Mode (|||48758|)".as_bytes());
assert_eq!(port_epsv_nodot, Ok((&b""[..], 48758)));
}
#[test]
fn test_active_eprt_valid() {
let port =
ftp_active_eprt("EPRT |2|2a01:e34:ee97:b130:8c3e:45ea:5ac6:e301|41813|".as_bytes());
assert_eq!(port, Ok((&b""[..], 41813)));
}
#[test]
fn test_active_port_valid() {
let port = ftp_active_port("PORT 192,168,0,13,234,10".as_bytes());
assert_eq!(port, Ok((&b""[..], 59914)));
}
#[test]
fn test_pasv_response_too_large() {
let port =
ftp_pasv_response("227 Entering Passive Mode (212,27,32,66,257,243).".as_bytes());
assert!(port.is_err());
let port =
ftp_pasv_response("227 Entering Passive Mode (212,27,32,66,255,65535).".as_bytes());
assert!(port.is_err());
}
#[test]
fn test_active_eprt_too_large() {
let port =
ftp_active_eprt("EPRT |2|2a01:e34:ee97:b130:8c3e:45ea:5ac6:e301|81813|".as_bytes());
assert!(port.is_err());
}
#[test]
fn test_active_port_too_large() {
let port = ftp_active_port("PORT 212,27,32,66,257,243".as_bytes());
assert!(port.is_err());
let port = ftp_active_port("PORT 212,27,32,66,255,65535".as_bytes());
assert!(port.is_err());
}
}