massive cleanup
This commit is contained in:
parent
4f82f07236
commit
1542492e61
338
src/nat/mod.rs
338
src/nat/mod.rs
@ -1,14 +1,14 @@
|
|||||||
use self::{
|
use crate::packet::protocols::{
|
||||||
// interface::Nat64Interface,
|
icmp::IcmpPacket, icmpv6::Icmpv6Packet, ipv4::Ipv4Packet, ipv6::Ipv6Packet, raw::RawBytes,
|
||||||
packet::IpPacket,
|
tcp::TcpPacket, udp::UdpPacket,
|
||||||
table::{Nat64Table, TableError},
|
|
||||||
};
|
};
|
||||||
use crate::{
|
|
||||||
into_icmp, into_icmpv6, into_tcp, into_udp, ipv4_packet, ipv6_packet,
|
use self::{
|
||||||
nat::xlat::translate_udp_4_to_6,
|
table::Nat64Table,
|
||||||
|
utils::{embed_address, extract_address},
|
||||||
};
|
};
|
||||||
use ipnet::{Ipv4Net, Ipv6Net};
|
use ipnet::{Ipv4Net, Ipv6Net};
|
||||||
use pnet_packet::{ip::IpNextHeaderProtocols, Packet};
|
use pnet_packet::ip::IpNextHeaderProtocols;
|
||||||
use protomask_tun::TunDevice;
|
use protomask_tun::TunDevice;
|
||||||
use std::{
|
use std::{
|
||||||
net::{IpAddr, Ipv4Addr, Ipv6Addr},
|
net::{IpAddr, Ipv4Addr, Ipv6Addr},
|
||||||
@ -18,8 +18,9 @@ use tokio::sync::{broadcast, mpsc};
|
|||||||
|
|
||||||
mod interface;
|
mod interface;
|
||||||
mod macros;
|
mod macros;
|
||||||
mod packet;
|
// mod packet;
|
||||||
mod table;
|
mod table;
|
||||||
|
mod utils;
|
||||||
mod xlat;
|
mod xlat;
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error)]
|
#[derive(Debug, thiserror::Error)]
|
||||||
@ -33,6 +34,8 @@ pub enum Nat64Error {
|
|||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
XlatError(#[from] xlat::PacketTranslationError),
|
XlatError(#[from] xlat::PacketTranslationError),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
PacketHandlingError(#[from] crate::packet::error::PacketError),
|
||||||
|
#[error(transparent)]
|
||||||
PacketReceiveError(#[from] broadcast::error::RecvError),
|
PacketReceiveError(#[from] broadcast::error::RecvError),
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,219 +87,130 @@ impl Nat64 {
|
|||||||
// Try to read a packet
|
// Try to read a packet
|
||||||
let packet = rx.recv().await?;
|
let packet = rx.recv().await?;
|
||||||
|
|
||||||
|
// Clone the TX so the worker can respond with data
|
||||||
|
|
||||||
// Spawn a task to process the packet
|
|
||||||
let mut tx = tx.clone();
|
let mut tx = tx.clone();
|
||||||
tokio::spawn(async move{
|
|
||||||
|
|
||||||
|
// Separate logic is needed for handling IPv4 vs IPv6 packets, so a check must be done here
|
||||||
|
match packet[0] >> 4 {
|
||||||
|
4 => {
|
||||||
|
// Parse the packet
|
||||||
|
let packet: Ipv4Packet<Vec<u8>> = packet.try_into()?;
|
||||||
|
|
||||||
});
|
// Drop packets that aren't destined for a destination the table knows about
|
||||||
|
if !self.table.contains(&IpAddr::V4(packet.destination_address)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the new source and dest addresses
|
||||||
|
let new_source = embed_address(packet.source_address, self.ipv6_nat_prefix);
|
||||||
|
let new_destination = self.table.get_reverse(packet.destination_address)?;
|
||||||
|
|
||||||
|
// Spawn a task to process the packet
|
||||||
|
tokio::spawn(async move {
|
||||||
|
process_inbound_ipv4(packet, new_source, new_destination, &mut tx)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
6 => {
|
||||||
|
// Parse the packet
|
||||||
|
let packet: Ipv6Packet<Vec<u8>> = packet.try_into()?;
|
||||||
|
|
||||||
|
// Get the new source and dest addresses
|
||||||
|
let new_source = self.table.get_or_assign_ipv4(packet.source_address)?;
|
||||||
|
let new_destination = extract_address(packet.destination_address);
|
||||||
|
|
||||||
|
// Spawn a task to process the packet
|
||||||
|
tokio::spawn(async move {
|
||||||
|
process_inbound_ipv6(packet, new_source, new_destination, &mut tx)
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
n => {
|
||||||
|
log::warn!("Unknown IP version: {}", n);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
// // Allocate a buffer for incoming packets
|
|
||||||
// let mut buffer = vec![0u8; self.interface.mtu()];
|
|
||||||
|
|
||||||
// // Loop forever
|
|
||||||
// loop {
|
|
||||||
// // Read a packet from the interface
|
|
||||||
// match self.interface.recv(&mut buffer) {
|
|
||||||
// Ok(packet_len) => {
|
|
||||||
// // Parse in to a more friendly format
|
|
||||||
// log::debug!("--- NEW PACKET ---");
|
|
||||||
// match IpPacket::new(&buffer[..packet_len]) {
|
|
||||||
// // Try to process the packet
|
|
||||||
// Ok(inbound_packet) => match self.process_packet(inbound_packet).await {
|
|
||||||
// Ok(outbound_packet) => match outbound_packet {
|
|
||||||
// // If data is returned, send it back out the interface
|
|
||||||
// Some(outbound_packet) => {
|
|
||||||
// let packet_bytes = outbound_packet.to_bytes();
|
|
||||||
// log::debug!(
|
|
||||||
// "Outbound packet next header: {}",
|
|
||||||
// outbound_packet.get_next_header().0
|
|
||||||
// );
|
|
||||||
// log::debug!("Sending packet: {:?}", packet_bytes);
|
|
||||||
// self.interface.send(&packet_bytes).unwrap();
|
|
||||||
// }
|
|
||||||
// // Otherwise, we can assume that the packet was dealt with, and can move on
|
|
||||||
// None => {}
|
|
||||||
// },
|
|
||||||
|
|
||||||
// // Some errors are non-critical as far as this loop is concerned
|
|
||||||
// Err(error) => match error {
|
|
||||||
// Nat64Error::TableError(TableError::NoIpv6Mapping(address)) => {
|
|
||||||
// log::debug!("No IPv6 mapping for {}", address);
|
|
||||||
// }
|
|
||||||
// error => {
|
|
||||||
// return Err(error);
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
// },
|
|
||||||
// Err(error) => {
|
|
||||||
// log::error!("Failed to parse packet: {}", error);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// Err(error) => {
|
|
||||||
// log::error!("Failed to read packet: {}", error);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Nat64 {
|
/// Process an inbound IPv4 packet
|
||||||
|
async fn process_inbound_ipv4(
|
||||||
async fn process_packet<'a>(
|
packet: Ipv4Packet<Vec<u8>>,
|
||||||
&mut self,
|
new_source: Ipv6Addr,
|
||||||
packet: IpPacket<'a>,
|
new_destination: Ipv6Addr,
|
||||||
) -> Result<Option<IpPacket<'a>>, Nat64Error> {
|
tx: &mut mpsc::Sender<Vec<u8>>,
|
||||||
// The destination of the packet must be within a prefix we care about
|
) -> Result<(), Nat64Error> {
|
||||||
if match packet.get_destination() {
|
// Handle each possible embedded packet type
|
||||||
IpAddr::V4(ipv4_addr) => !self.table.is_address_within_pool(&ipv4_addr),
|
match packet.protocol {
|
||||||
IpAddr::V6(ipv6_addr) => !self.ipv6_nat_prefix.contains(&ipv6_addr),
|
IpNextHeaderProtocols::Icmp => {
|
||||||
} {
|
let icmp_packet: IcmpPacket<RawBytes> = packet.payload.try_into()?;
|
||||||
log::debug!(
|
todo!()
|
||||||
"Packet destination {} is not within the NAT64 prefix or IPv4 pool",
|
|
||||||
packet.get_destination(),
|
|
||||||
);
|
|
||||||
return Ok(None);
|
|
||||||
}
|
}
|
||||||
|
IpNextHeaderProtocols::Udp => {
|
||||||
// Compute the translated source and dest addresses
|
let udp_packet: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
|
||||||
let source = packet.get_source();
|
&packet.payload,
|
||||||
let new_source = self
|
IpAddr::V4(packet.source_address),
|
||||||
.table
|
IpAddr::V4(packet.destination_address),
|
||||||
.calculate_xlat_addr(&source, &self.ipv6_nat_prefix)?;
|
)?;
|
||||||
let destination = packet.get_destination();
|
todo!()
|
||||||
let new_destination = self
|
}
|
||||||
.table
|
IpNextHeaderProtocols::Tcp => {
|
||||||
.calculate_xlat_addr(&destination, &self.ipv6_nat_prefix)?;
|
let tcp_packet: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
|
||||||
|
&packet.payload,
|
||||||
// Log information about the packet
|
IpAddr::V4(packet.source_address),
|
||||||
log::debug!(
|
IpAddr::V4(packet.destination_address),
|
||||||
"Received packet traveling from {} to {}",
|
)?;
|
||||||
source,
|
todo!()
|
||||||
destination
|
}
|
||||||
);
|
_ => {
|
||||||
log::debug!(
|
log::warn!("Unsupported next level protocol: {}", packet.protocol);
|
||||||
"New path shall become: {} -> {}",
|
|
||||||
new_source,
|
|
||||||
new_destination
|
|
||||||
);
|
|
||||||
|
|
||||||
// Different logic is required for ICMP, UDP, and TCP
|
|
||||||
match (packet, new_source, new_destination) {
|
|
||||||
(IpPacket::V4(packet), IpAddr::V6(new_source), IpAddr::V6(new_destination)) => {
|
|
||||||
match packet.get_next_level_protocol() {
|
|
||||||
// Internet Control Message Protocol
|
|
||||||
IpNextHeaderProtocols::Icmp => match xlat::translate_icmp_4_to_6(
|
|
||||||
into_icmp!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
)? {
|
|
||||||
Some(icmp_packet) => Ok(Some(IpPacket::V6(ipv6_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Icmpv6,
|
|
||||||
packet.get_ttl(),
|
|
||||||
icmp_packet.packet()
|
|
||||||
)))),
|
|
||||||
None => Ok(None),
|
|
||||||
},
|
|
||||||
|
|
||||||
// User Datagram Protocol
|
|
||||||
IpNextHeaderProtocols::Udp => Ok(Some(IpPacket::V6(ipv6_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Udp,
|
|
||||||
packet.get_ttl(),
|
|
||||||
translate_udp_4_to_6(
|
|
||||||
into_udp!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination
|
|
||||||
)?
|
|
||||||
.packet()
|
|
||||||
)))),
|
|
||||||
|
|
||||||
// Transmission Control Protocol
|
|
||||||
IpNextHeaderProtocols::Tcp => Ok(Some(IpPacket::V6(ipv6_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Tcp,
|
|
||||||
packet.get_ttl(),
|
|
||||||
xlat::translate_tcp_4_to_6(
|
|
||||||
into_tcp!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination
|
|
||||||
)?
|
|
||||||
.packet()
|
|
||||||
)))),
|
|
||||||
|
|
||||||
// For any protocol we don't support, just warn and drop the packet
|
|
||||||
next_level_protocol => {
|
|
||||||
log::warn!("Unsupported next level protocol: {}", next_level_protocol);
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
(IpPacket::V6(packet), IpAddr::V4(new_source), IpAddr::V4(new_destination)) => {
|
|
||||||
match packet.get_next_header() {
|
|
||||||
// Internet Control Message Protocol Version 6
|
|
||||||
IpNextHeaderProtocols::Icmpv6 => match xlat::translate_icmp_6_to_4(
|
|
||||||
into_icmpv6!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
)? {
|
|
||||||
Some(icmp_packet) => Ok(Some(IpPacket::V4(ipv4_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Icmp,
|
|
||||||
packet.get_hop_limit(),
|
|
||||||
icmp_packet.packet()
|
|
||||||
)))),
|
|
||||||
None => Ok(None),
|
|
||||||
},
|
|
||||||
|
|
||||||
// User Datagram Protocol
|
|
||||||
IpNextHeaderProtocols::Udp => Ok(Some(IpPacket::V4(ipv4_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Udp,
|
|
||||||
packet.get_hop_limit(),
|
|
||||||
xlat::translate_udp_6_to_4(
|
|
||||||
into_udp!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination
|
|
||||||
)?
|
|
||||||
.packet()
|
|
||||||
)))),
|
|
||||||
|
|
||||||
// Transmission Control Protocol
|
|
||||||
IpNextHeaderProtocols::Tcp => Ok(Some(IpPacket::V4(ipv4_packet!(
|
|
||||||
new_source,
|
|
||||||
new_destination,
|
|
||||||
IpNextHeaderProtocols::Tcp,
|
|
||||||
packet.get_hop_limit(),
|
|
||||||
xlat::translate_tcp_6_to_4(
|
|
||||||
into_tcp!(packet.payload().to_vec())?,
|
|
||||||
new_source,
|
|
||||||
new_destination
|
|
||||||
)?
|
|
||||||
.packet()
|
|
||||||
)))),
|
|
||||||
|
|
||||||
// For any protocol we don't support, just warn and drop the packet
|
|
||||||
next_header_protocol => {
|
|
||||||
log::warn!("Unsupported next header protocol: {}", next_header_protocol);
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Honestly, this should probably be `unreachable!()`
|
|
||||||
_ => unimplemented!(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Process an inbound IPv6 packet
|
||||||
|
async fn process_inbound_ipv6(
|
||||||
|
packet: Ipv6Packet<Vec<u8>>,
|
||||||
|
new_source: Ipv4Addr,
|
||||||
|
new_destination: Ipv4Addr,
|
||||||
|
tx: &mut mpsc::Sender<Vec<u8>>,
|
||||||
|
) -> Result<(), Nat64Error> {
|
||||||
|
// Handle each possible embedded packet type
|
||||||
|
match packet.next_header {
|
||||||
|
IpNextHeaderProtocols::Icmpv6 => {
|
||||||
|
let icmpv6_packet: Icmpv6Packet<RawBytes> = Icmpv6Packet::new_from_bytes_raw_payload(
|
||||||
|
&packet.payload,
|
||||||
|
packet.source_address,
|
||||||
|
packet.destination_address,
|
||||||
|
)?;
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
IpNextHeaderProtocols::Udp => {
|
||||||
|
let udp_packet: UdpPacket<RawBytes> = UdpPacket::new_from_bytes_raw_payload(
|
||||||
|
&packet.payload,
|
||||||
|
IpAddr::V6(packet.source_address),
|
||||||
|
IpAddr::V6(packet.destination_address),
|
||||||
|
)?;
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
IpNextHeaderProtocols::Tcp => {
|
||||||
|
let tcp_packet: TcpPacket<RawBytes> = TcpPacket::new_from_bytes_raw_payload(
|
||||||
|
&packet.payload,
|
||||||
|
IpAddr::V6(packet.source_address),
|
||||||
|
IpAddr::V6(packet.destination_address),
|
||||||
|
)?;
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
log::warn!("Unsupported next level protocol: {}", packet.next_header);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -1,159 +1,159 @@
|
|||||||
//! A generic internet protocol packet type
|
// //! A generic internet protocol packet type
|
||||||
|
|
||||||
use std::net::IpAddr;
|
// use std::net::IpAddr;
|
||||||
|
|
||||||
use pnet_packet::{ip::IpNextHeaderProtocol, ipv4::Ipv4Packet, ipv6::Ipv6Packet, Packet};
|
// use pnet_packet::{ip::IpNextHeaderProtocol, ipv4::Ipv4Packet, ipv6::Ipv6Packet, Packet};
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error)]
|
// #[derive(Debug, thiserror::Error)]
|
||||||
pub enum PacketError {
|
// pub enum PacketError {
|
||||||
#[error("Packed too small (len: {0})")]
|
// #[error("Packed too small (len: {0})")]
|
||||||
PacketTooSmall(usize),
|
// PacketTooSmall(usize),
|
||||||
#[error("Unknown Internet Protocol version: {0}")]
|
// #[error("Unknown Internet Protocol version: {0}")]
|
||||||
UnknownVersion(u8),
|
// UnknownVersion(u8),
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// A protocol-agnostic packet type
|
// /// A protocol-agnostic packet type
|
||||||
#[derive(Debug)]
|
// #[derive(Debug)]
|
||||||
pub enum IpPacket<'a> {
|
// pub enum IpPacket<'a> {
|
||||||
/// IPv4 packet
|
// /// IPv4 packet
|
||||||
V4(Ipv4Packet<'a>),
|
// V4(Ipv4Packet<'a>),
|
||||||
/// IPv6 packet
|
// /// IPv6 packet
|
||||||
V6(Ipv6Packet<'a>),
|
// V6(Ipv6Packet<'a>),
|
||||||
}
|
// }
|
||||||
|
|
||||||
impl IpPacket<'_> {
|
// impl IpPacket<'_> {
|
||||||
/// Creates a new packet from a byte slice
|
// /// Creates a new packet from a byte slice
|
||||||
pub fn new<'a>(bytes: &'a [u8]) -> Result<IpPacket<'a>, PacketError> {
|
// pub fn new<'a>(bytes: &'a [u8]) -> Result<IpPacket<'a>, PacketError> {
|
||||||
// Parse the packet. If there is an error, cast None to the error type
|
// // Parse the packet. If there is an error, cast None to the error type
|
||||||
match bytes[0] >> 4 {
|
// match bytes[0] >> 4 {
|
||||||
4 => Ok(IpPacket::V4(
|
// 4 => Ok(IpPacket::V4(
|
||||||
Ipv4Packet::new(bytes).ok_or_else(|| PacketError::PacketTooSmall(bytes.len()))?,
|
// Ipv4Packet::new(bytes).ok_or_else(|| PacketError::PacketTooSmall(bytes.len()))?,
|
||||||
)),
|
// )),
|
||||||
6 => Ok(IpPacket::V6(
|
// 6 => Ok(IpPacket::V6(
|
||||||
Ipv6Packet::new(bytes).ok_or_else(|| PacketError::PacketTooSmall(bytes.len()))?,
|
// Ipv6Packet::new(bytes).ok_or_else(|| PacketError::PacketTooSmall(bytes.len()))?,
|
||||||
)),
|
// )),
|
||||||
n => Err(PacketError::UnknownVersion(n)),
|
// n => Err(PacketError::UnknownVersion(n)),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Returns the source address
|
// /// Returns the source address
|
||||||
pub fn get_source(&self) -> IpAddr {
|
// pub fn get_source(&self) -> IpAddr {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => IpAddr::V4(packet.get_source()),
|
// IpPacket::V4(packet) => IpAddr::V4(packet.get_source()),
|
||||||
IpPacket::V6(packet) => IpAddr::V6(packet.get_source()),
|
// IpPacket::V6(packet) => IpAddr::V6(packet.get_source()),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Returns the destination address
|
// /// Returns the destination address
|
||||||
pub fn get_destination(&self) -> IpAddr {
|
// pub fn get_destination(&self) -> IpAddr {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => IpAddr::V4(packet.get_destination()),
|
// IpPacket::V4(packet) => IpAddr::V4(packet.get_destination()),
|
||||||
IpPacket::V6(packet) => IpAddr::V6(packet.get_destination()),
|
// IpPacket::V6(packet) => IpAddr::V6(packet.get_destination()),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Returns the packet header
|
// /// Returns the packet header
|
||||||
#[allow(dead_code)]
|
// #[allow(dead_code)]
|
||||||
pub fn get_header(&self) -> &[u8] {
|
// pub fn get_header(&self) -> &[u8] {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.packet()[..20].as_ref(),
|
// IpPacket::V4(packet) => packet.packet()[..20].as_ref(),
|
||||||
IpPacket::V6(packet) => packet.packet()[..40].as_ref(),
|
// IpPacket::V6(packet) => packet.packet()[..40].as_ref(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Returns the packet payload
|
// /// Returns the packet payload
|
||||||
#[allow(dead_code)]
|
// #[allow(dead_code)]
|
||||||
pub fn get_payload(&self) -> &[u8] {
|
// pub fn get_payload(&self) -> &[u8] {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.payload(),
|
// IpPacket::V4(packet) => packet.payload(),
|
||||||
IpPacket::V6(packet) => packet.payload(),
|
// IpPacket::V6(packet) => packet.payload(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Converts the packet to a byte vector
|
// /// Converts the packet to a byte vector
|
||||||
pub fn to_bytes(&self) -> Vec<u8> {
|
// pub fn to_bytes(&self) -> Vec<u8> {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.packet().to_vec(),
|
// IpPacket::V4(packet) => packet.packet().to_vec(),
|
||||||
IpPacket::V6(packet) => packet.packet().to_vec(),
|
// IpPacket::V6(packet) => packet.packet().to_vec(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Returns the packet length
|
// /// Returns the packet length
|
||||||
#[allow(dead_code)]
|
// #[allow(dead_code)]
|
||||||
pub fn len(&self) -> usize {
|
// pub fn len(&self) -> usize {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.packet().len(),
|
// IpPacket::V4(packet) => packet.packet().len(),
|
||||||
IpPacket::V6(packet) => packet.packet().len(),
|
// IpPacket::V6(packet) => packet.packet().len(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Get the next header
|
// /// Get the next header
|
||||||
pub fn get_next_header(&self) -> IpNextHeaderProtocol {
|
// pub fn get_next_header(&self) -> IpNextHeaderProtocol {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.get_next_level_protocol(),
|
// IpPacket::V4(packet) => packet.get_next_level_protocol(),
|
||||||
IpPacket::V6(packet) => packet.get_next_header(),
|
// IpPacket::V6(packet) => packet.get_next_header(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
/// Get the TTL
|
// /// Get the TTL
|
||||||
#[allow(dead_code)]
|
// #[allow(dead_code)]
|
||||||
pub fn get_ttl(&self) -> u8 {
|
// pub fn get_ttl(&self) -> u8 {
|
||||||
match self {
|
// match self {
|
||||||
IpPacket::V4(packet) => packet.get_ttl(),
|
// IpPacket::V4(packet) => packet.get_ttl(),
|
||||||
IpPacket::V6(packet) => packet.get_hop_limit(),
|
// IpPacket::V6(packet) => packet.get_hop_limit(),
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
#[cfg(test)]
|
// #[cfg(test)]
|
||||||
mod tests {
|
// mod tests {
|
||||||
use pnet_packet::{ipv4::MutableIpv4Packet, ipv6::MutableIpv6Packet};
|
// use pnet_packet::{ipv4::MutableIpv4Packet, ipv6::MutableIpv6Packet};
|
||||||
|
|
||||||
use super::*;
|
// use super::*;
|
||||||
|
|
||||||
#[test]
|
// #[test]
|
||||||
fn test_ipv4_packet() {
|
// fn test_ipv4_packet() {
|
||||||
// Build packet to test
|
// // Build packet to test
|
||||||
let mut packet = MutableIpv4Packet::owned(vec![0; 20]).unwrap();
|
// let mut packet = MutableIpv4Packet::owned(vec![0; 20]).unwrap();
|
||||||
packet.set_version(4);
|
// packet.set_version(4);
|
||||||
packet.set_source("192.0.2.1".parse().unwrap());
|
// packet.set_source("192.0.2.1".parse().unwrap());
|
||||||
packet.set_destination("192.0.2.2".parse().unwrap());
|
// packet.set_destination("192.0.2.2".parse().unwrap());
|
||||||
|
|
||||||
// Parse
|
// // Parse
|
||||||
let header = packet.packet()[..20].to_vec();
|
// let header = packet.packet()[..20].to_vec();
|
||||||
let packet = IpPacket::new(packet.packet()).unwrap();
|
// let packet = IpPacket::new(packet.packet()).unwrap();
|
||||||
assert_eq!(
|
// assert_eq!(
|
||||||
packet.get_source(),
|
// packet.get_source(),
|
||||||
IpAddr::V4("192.0.2.1".parse().unwrap())
|
// IpAddr::V4("192.0.2.1".parse().unwrap())
|
||||||
);
|
// );
|
||||||
assert_eq!(
|
// assert_eq!(
|
||||||
packet.get_destination(),
|
// packet.get_destination(),
|
||||||
IpAddr::V4("192.0.2.2".parse().unwrap())
|
// IpAddr::V4("192.0.2.2".parse().unwrap())
|
||||||
);
|
// );
|
||||||
assert_eq!(packet.get_header(), header);
|
// assert_eq!(packet.get_header(), header);
|
||||||
}
|
// }
|
||||||
|
|
||||||
#[test]
|
// #[test]
|
||||||
fn test_ipv6_packet() {
|
// fn test_ipv6_packet() {
|
||||||
// Build packet to test
|
// // Build packet to test
|
||||||
let mut packet = MutableIpv6Packet::owned(vec![0; 40]).unwrap();
|
// let mut packet = MutableIpv6Packet::owned(vec![0; 40]).unwrap();
|
||||||
packet.set_version(6);
|
// packet.set_version(6);
|
||||||
packet.set_source("2001:db8::c0a8:1".parse().unwrap());
|
// packet.set_source("2001:db8::c0a8:1".parse().unwrap());
|
||||||
packet.set_destination("2001:db8::c0a8:2".parse().unwrap());
|
// packet.set_destination("2001:db8::c0a8:2".parse().unwrap());
|
||||||
|
|
||||||
// Parse
|
// // Parse
|
||||||
let header = packet.packet()[..40].to_vec();
|
// let header = packet.packet()[..40].to_vec();
|
||||||
let packet = IpPacket::new(packet.packet()).unwrap();
|
// let packet = IpPacket::new(packet.packet()).unwrap();
|
||||||
|
|
||||||
// Test
|
// // Test
|
||||||
assert_eq!(
|
// assert_eq!(
|
||||||
packet.get_source(),
|
// packet.get_source(),
|
||||||
IpAddr::V6("2001:db8::c0a8:1".parse().unwrap())
|
// IpAddr::V6("2001:db8::c0a8:1".parse().unwrap())
|
||||||
);
|
// );
|
||||||
assert_eq!(
|
// assert_eq!(
|
||||||
packet.get_destination(),
|
// packet.get_destination(),
|
||||||
IpAddr::V6("2001:db8::c0a8:2".parse().unwrap())
|
// IpAddr::V6("2001:db8::c0a8:2".parse().unwrap())
|
||||||
);
|
// );
|
||||||
assert_eq!(packet.get_header(), header);
|
// assert_eq!(packet.get_header(), header);
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
@ -67,6 +67,14 @@ impl Nat64Table {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Check if a given address exists in the table
|
||||||
|
pub fn contains(&self, address: &IpAddr) -> bool {
|
||||||
|
match address {
|
||||||
|
IpAddr::V4(ipv4) => self.reservations.contains_right(ipv4),
|
||||||
|
IpAddr::V6(ipv6) => self.reservations.contains_left(ipv6),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Get or assign an IPv4 address for the given IPv6 address
|
/// Get or assign an IPv4 address for the given IPv6 address
|
||||||
pub fn get_or_assign_ipv4(&mut self, ipv6: Ipv6Addr) -> Result<Ipv4Addr, TableError> {
|
pub fn get_or_assign_ipv4(&mut self, ipv6: Ipv6Addr) -> Result<Ipv4Addr, TableError> {
|
||||||
// Prune old reservations
|
// Prune old reservations
|
||||||
|
25
src/nat/utils.rs
Normal file
25
src/nat/utils.rs
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
use std::net::{Ipv4Addr, Ipv6Addr};
|
||||||
|
|
||||||
|
use ipnet::Ipv6Net;
|
||||||
|
|
||||||
|
/// Embed an IPv4 address in an IPv6 prefix
|
||||||
|
pub fn embed_address(ipv4_address: Ipv4Addr, ipv6_prefix: Ipv6Net) -> Ipv6Addr {
|
||||||
|
let v4_octets = ipv4_address.octets();
|
||||||
|
let v6_octets = ipv6_prefix.addr().octets();
|
||||||
|
Ipv6Addr::new(
|
||||||
|
u16::from_be_bytes([v6_octets[0], v6_octets[1]]),
|
||||||
|
u16::from_be_bytes([v6_octets[2], v6_octets[3]]),
|
||||||
|
u16::from_be_bytes([v6_octets[4], v6_octets[5]]),
|
||||||
|
u16::from_be_bytes([v6_octets[6], v6_octets[7]]),
|
||||||
|
u16::from_be_bytes([v6_octets[8], v6_octets[9]]),
|
||||||
|
u16::from_be_bytes([v6_octets[10], v6_octets[11]]),
|
||||||
|
u16::from_be_bytes([v4_octets[0], v4_octets[1]]),
|
||||||
|
u16::from_be_bytes([v4_octets[2], v4_octets[3]]),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Extract an IPv4 address from an IPv6 address
|
||||||
|
pub fn extract_address(ipv6_address: Ipv6Addr) -> Ipv4Addr {
|
||||||
|
let octets = ipv6_address.octets();
|
||||||
|
Ipv4Addr::new(octets[12], octets[13], octets[14], octets[15])
|
||||||
|
}
|
@ -43,6 +43,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl<T> Into<Vec<u8>> for IcmpPacket<T>
|
impl<T> Into<Vec<u8>> for IcmpPacket<T>
|
||||||
where
|
where
|
||||||
T: Into<Vec<u8>>,
|
T: Into<Vec<u8>>,
|
||||||
|
@ -7,6 +7,8 @@ use pnet_packet::{
|
|||||||
|
|
||||||
use crate::packet::error::PacketError;
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
|
use super::raw::RawBytes;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
pub struct Icmpv6Packet<T> {
|
pub struct Icmpv6Packet<T> {
|
||||||
pub source_address: Ipv6Addr,
|
pub source_address: Ipv6Addr,
|
||||||
@ -60,7 +62,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Icmpv6Packet<Vec<u8>> {
|
impl Icmpv6Packet<RawBytes> {
|
||||||
/// Construct a new ICMPv6 packet with a raw payload from raw bytes
|
/// Construct a new ICMPv6 packet with a raw payload from raw bytes
|
||||||
pub fn new_from_bytes_raw_payload(
|
pub fn new_from_bytes_raw_payload(
|
||||||
bytes: &[u8],
|
bytes: &[u8],
|
||||||
@ -77,7 +79,7 @@ impl Icmpv6Packet<Vec<u8>> {
|
|||||||
destination_address,
|
destination_address,
|
||||||
icmp_type: packet.get_icmpv6_type(),
|
icmp_type: packet.get_icmpv6_type(),
|
||||||
icmp_code: packet.get_icmpv6_code(),
|
icmp_code: packet.get_icmpv6_code(),
|
||||||
payload: packet.payload().to_vec(),
|
payload: RawBytes(packet.payload().to_vec()),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,5 +2,6 @@ pub mod icmp;
|
|||||||
pub mod icmpv6;
|
pub mod icmpv6;
|
||||||
pub mod ipv4;
|
pub mod ipv4;
|
||||||
pub mod ipv6;
|
pub mod ipv6;
|
||||||
|
pub mod raw;
|
||||||
pub mod tcp;
|
pub mod tcp;
|
||||||
pub mod udp;
|
pub mod udp;
|
||||||
|
12
src/packet/protocols/raw.rs
Normal file
12
src/packet/protocols/raw.rs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
||||||
|
pub struct RawBytes(pub Vec<u8>);
|
||||||
|
|
||||||
|
impl TryFrom<Vec<u8>> for RawBytes {
|
||||||
|
type Error = PacketError;
|
||||||
|
|
||||||
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
|
Ok(Self(bytes))
|
||||||
|
}
|
||||||
|
}
|
@ -5,6 +5,7 @@ use pnet_packet::{
|
|||||||
Packet,
|
Packet,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use super::raw::RawBytes;
|
||||||
use crate::packet::error::PacketError;
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
/// A TCP packet
|
/// A TCP packet
|
||||||
@ -145,7 +146,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TcpPacket<Vec<u8>> {
|
impl TcpPacket<RawBytes> {
|
||||||
/// Construct a new TCP packet with a raw payload from bytes
|
/// Construct a new TCP packet with a raw payload from bytes
|
||||||
pub fn new_from_bytes_raw_payload(
|
pub fn new_from_bytes_raw_payload(
|
||||||
bytes: &[u8],
|
bytes: &[u8],
|
||||||
@ -174,7 +175,7 @@ impl TcpPacket<Vec<u8>> {
|
|||||||
window_size: parsed.get_window(),
|
window_size: parsed.get_window(),
|
||||||
urgent_pointer: parsed.get_urgent_ptr(),
|
urgent_pointer: parsed.get_urgent_ptr(),
|
||||||
options: parsed.get_options().to_vec(),
|
options: parsed.get_options().to_vec(),
|
||||||
payload: parsed.payload().to_vec(),
|
payload: RawBytes(parsed.payload().to_vec()),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ use std::net::{IpAddr, SocketAddr};
|
|||||||
|
|
||||||
use pnet_packet::Packet;
|
use pnet_packet::Packet;
|
||||||
|
|
||||||
|
use super::raw::RawBytes;
|
||||||
use crate::packet::error::PacketError;
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
/// A UDP packet
|
/// A UDP packet
|
||||||
@ -109,7 +110,7 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl UdpPacket<Vec<u8>> {
|
impl UdpPacket<RawBytes> {
|
||||||
/// Construct a new UDP packet with a raw payload from bytes
|
/// Construct a new UDP packet with a raw payload from bytes
|
||||||
pub fn new_from_bytes_raw_payload(
|
pub fn new_from_bytes_raw_payload(
|
||||||
bytes: &[u8],
|
bytes: &[u8],
|
||||||
@ -132,7 +133,7 @@ impl UdpPacket<Vec<u8>> {
|
|||||||
Ok(Self {
|
Ok(Self {
|
||||||
source: SocketAddr::new(source_address, parsed.get_source()),
|
source: SocketAddr::new(source_address, parsed.get_source()),
|
||||||
destination: SocketAddr::new(destination_address, parsed.get_destination()),
|
destination: SocketAddr::new(destination_address, parsed.get_destination()),
|
||||||
payload: parsed.payload().to_vec(),
|
payload: RawBytes(parsed.payload().to_vec()),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user