Add ipv4
This commit is contained in:
parent
32a254a4ce
commit
4f82f07236
@ -23,37 +23,22 @@ impl<T> IcmpPacket<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> IcmpPacket<T>
|
impl<T> TryFrom<Vec<u8>> for IcmpPacket<T>
|
||||||
where
|
where
|
||||||
T: From<Vec<u8>>,
|
T: TryFrom<Vec<u8>, Error = PacketError>,
|
||||||
{
|
{
|
||||||
/// Construct a new ICMPv6 packet from raw bytes
|
type Error = PacketError;
|
||||||
pub fn new_from_bytes(bytes: &[u8]) -> Result<Self, PacketError> {
|
|
||||||
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let packet =
|
let packet =
|
||||||
pnet_packet::icmp::IcmpPacket::new(bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
pnet_packet::icmp::IcmpPacket::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
icmp_type: packet.get_icmp_type(),
|
icmp_type: packet.get_icmp_type(),
|
||||||
icmp_code: packet.get_icmp_code(),
|
icmp_code: packet.get_icmp_code(),
|
||||||
payload: packet.payload().to_vec().into(),
|
payload: packet.payload().to_vec().try_into()?,
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl IcmpPacket<Vec<u8>> {
|
|
||||||
/// Construct a new ICMPv6 packet with a raw payload from raw bytes
|
|
||||||
pub fn new_from_bytes_raw_payload(bytes: &[u8]) -> Result<Self, PacketError> {
|
|
||||||
// Parse the packet
|
|
||||||
let packet =
|
|
||||||
pnet_packet::icmp::IcmpPacket::new(bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
|
||||||
|
|
||||||
// Return the packet
|
|
||||||
Ok(Self {
|
|
||||||
icmp_type: packet.get_icmp_type(),
|
|
||||||
icmp_code: packet.get_icmp_code(),
|
|
||||||
payload: packet.payload().to_vec(),
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
131
src/packet/protocols/ipv4.rs
Normal file
131
src/packet/protocols/ipv4.rs
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
use std::net::Ipv4Addr;
|
||||||
|
|
||||||
|
use pnet_packet::{
|
||||||
|
ip::IpNextHeaderProtocol,
|
||||||
|
ipv4::{Ipv4Option, Ipv4OptionPacket},
|
||||||
|
Packet,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct Ipv4Packet<T> {
|
||||||
|
pub dscp: u8,
|
||||||
|
pub ecn: u8,
|
||||||
|
pub identification: u16,
|
||||||
|
pub flags: u8,
|
||||||
|
pub fragment_offset: u16,
|
||||||
|
pub ttl: u8,
|
||||||
|
pub protocol: IpNextHeaderProtocol,
|
||||||
|
pub source_address: Ipv4Addr,
|
||||||
|
pub destination_address: Ipv4Addr,
|
||||||
|
pub options: Vec<Ipv4Option>,
|
||||||
|
pub payload: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Ipv4Packet<T> {
|
||||||
|
/// Construct a new IPv4 packet
|
||||||
|
pub fn new(
|
||||||
|
dscp: u8,
|
||||||
|
ecn: u8,
|
||||||
|
identification: u16,
|
||||||
|
flags: u8,
|
||||||
|
fragment_offset: u16,
|
||||||
|
ttl: u8,
|
||||||
|
protocol: IpNextHeaderProtocol,
|
||||||
|
source_address: Ipv4Addr,
|
||||||
|
destination_address: Ipv4Addr,
|
||||||
|
options: Vec<Ipv4Option>,
|
||||||
|
payload: T,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
dscp,
|
||||||
|
ecn,
|
||||||
|
identification,
|
||||||
|
flags,
|
||||||
|
fragment_offset,
|
||||||
|
ttl,
|
||||||
|
protocol,
|
||||||
|
source_address,
|
||||||
|
destination_address,
|
||||||
|
options,
|
||||||
|
payload,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn options_length_words(&self) -> u8 {
|
||||||
|
self.options
|
||||||
|
.iter()
|
||||||
|
.map(|option| Ipv4OptionPacket::packet_size(option) as u8)
|
||||||
|
.sum::<u8>()
|
||||||
|
/ 4
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> TryFrom<Vec<u8>> for Ipv4Packet<T>
|
||||||
|
where
|
||||||
|
T: From<Vec<u8>>,
|
||||||
|
{
|
||||||
|
type Error = PacketError;
|
||||||
|
|
||||||
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
|
// Parse the packet
|
||||||
|
let packet =
|
||||||
|
pnet_packet::ipv4::Ipv4Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
||||||
|
|
||||||
|
// Return the packet
|
||||||
|
Ok(Self {
|
||||||
|
dscp: packet.get_dscp(),
|
||||||
|
ecn: packet.get_ecn(),
|
||||||
|
identification: packet.get_identification(),
|
||||||
|
flags: packet.get_flags(),
|
||||||
|
fragment_offset: packet.get_fragment_offset(),
|
||||||
|
ttl: packet.get_ttl(),
|
||||||
|
protocol: packet.get_next_level_protocol(),
|
||||||
|
source_address: packet.get_source(),
|
||||||
|
destination_address: packet.get_destination(),
|
||||||
|
options: packet.get_options(),
|
||||||
|
payload: packet.payload().to_vec().into(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Into<Vec<u8>> for Ipv4Packet<T>
|
||||||
|
where
|
||||||
|
T: Into<Vec<u8>> + Clone,
|
||||||
|
{
|
||||||
|
fn into(self) -> Vec<u8> {
|
||||||
|
// Convert the payload into raw bytes
|
||||||
|
let payload: Vec<u8> = self.payload.clone().into();
|
||||||
|
|
||||||
|
// Build the packet
|
||||||
|
let total_length = 20 + (self.options_length_words() as usize * 4) + payload.len();
|
||||||
|
let mut packet =
|
||||||
|
pnet_packet::ipv4::MutableIpv4Packet::owned(vec![0u8; total_length]).unwrap();
|
||||||
|
|
||||||
|
// Set the fields
|
||||||
|
packet.set_version(4);
|
||||||
|
packet.set_header_length(5 + self.options_length_words());
|
||||||
|
packet.set_dscp(self.dscp);
|
||||||
|
packet.set_ecn(self.ecn);
|
||||||
|
packet.set_total_length(total_length.try_into().unwrap());
|
||||||
|
packet.set_identification(self.identification);
|
||||||
|
packet.set_flags(self.flags);
|
||||||
|
packet.set_fragment_offset(self.fragment_offset);
|
||||||
|
packet.set_ttl(self.ttl);
|
||||||
|
packet.set_next_level_protocol(self.protocol);
|
||||||
|
packet.set_source(self.source_address);
|
||||||
|
packet.set_destination(self.destination_address);
|
||||||
|
packet.set_options(&self.options);
|
||||||
|
|
||||||
|
// Set the payload
|
||||||
|
packet.set_payload(&payload);
|
||||||
|
|
||||||
|
// Calculate the checksum
|
||||||
|
packet.set_checksum(0);
|
||||||
|
packet.set_checksum(pnet_packet::ipv4::checksum(&packet.to_immutable()));
|
||||||
|
|
||||||
|
// Return the packet
|
||||||
|
packet.to_immutable().packet().to_vec()
|
||||||
|
}
|
||||||
|
}
|
95
src/packet/protocols/ipv6.rs
Normal file
95
src/packet/protocols/ipv6.rs
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
use std::net::Ipv6Addr;
|
||||||
|
|
||||||
|
use pnet_packet::{ip::IpNextHeaderProtocol, Packet};
|
||||||
|
|
||||||
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
||||||
|
pub struct Ipv6Packet<T> {
|
||||||
|
pub traffic_class: u8,
|
||||||
|
pub flow_label: u32,
|
||||||
|
pub next_header: IpNextHeaderProtocol,
|
||||||
|
pub hop_limit: u8,
|
||||||
|
pub source_address: Ipv6Addr,
|
||||||
|
pub destination_address: Ipv6Addr,
|
||||||
|
pub payload: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Ipv6Packet<T> {
|
||||||
|
/// Construct a new IPv6 packet
|
||||||
|
pub fn new(
|
||||||
|
traffic_class: u8,
|
||||||
|
flow_label: u32,
|
||||||
|
next_header: IpNextHeaderProtocol,
|
||||||
|
hop_limit: u8,
|
||||||
|
source_address: Ipv6Addr,
|
||||||
|
destination_address: Ipv6Addr,
|
||||||
|
payload: T,
|
||||||
|
) -> Self {
|
||||||
|
Self {
|
||||||
|
traffic_class,
|
||||||
|
flow_label,
|
||||||
|
next_header,
|
||||||
|
hop_limit,
|
||||||
|
source_address,
|
||||||
|
destination_address,
|
||||||
|
payload,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> TryFrom<Vec<u8>> for Ipv6Packet<T>
|
||||||
|
where
|
||||||
|
T: From<Vec<u8>>,
|
||||||
|
{
|
||||||
|
type Error = PacketError;
|
||||||
|
|
||||||
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
|
// Parse the packet
|
||||||
|
let packet =
|
||||||
|
pnet_packet::ipv6::Ipv6Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
||||||
|
|
||||||
|
// Return the packet
|
||||||
|
Ok(Self {
|
||||||
|
traffic_class: packet.get_traffic_class(),
|
||||||
|
flow_label: packet.get_flow_label(),
|
||||||
|
next_header: packet.get_next_header(),
|
||||||
|
hop_limit: packet.get_hop_limit(),
|
||||||
|
source_address: packet.get_source(),
|
||||||
|
destination_address: packet.get_destination(),
|
||||||
|
payload: packet.payload().to_vec().into(),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Into<Vec<u8>> for Ipv6Packet<T>
|
||||||
|
where
|
||||||
|
T: Into<Vec<u8>>,
|
||||||
|
{
|
||||||
|
fn into(self) -> Vec<u8> {
|
||||||
|
// Convert the payload into raw bytes
|
||||||
|
let payload: Vec<u8> = self.payload.into();
|
||||||
|
|
||||||
|
// Allocate a mutable packet to write into
|
||||||
|
let total_length =
|
||||||
|
pnet_packet::ipv6::MutableIpv6Packet::minimum_packet_size() + payload.len();
|
||||||
|
let mut output =
|
||||||
|
pnet_packet::ipv6::MutableIpv6Packet::owned(vec![0u8; total_length]).unwrap();
|
||||||
|
|
||||||
|
// Write the header
|
||||||
|
output.set_version(6);
|
||||||
|
output.set_traffic_class(self.traffic_class);
|
||||||
|
output.set_flow_label(self.flow_label);
|
||||||
|
output.set_payload_length(payload.len() as u16);
|
||||||
|
output.set_next_header(self.next_header);
|
||||||
|
output.set_hop_limit(self.hop_limit);
|
||||||
|
output.set_source(self.source_address);
|
||||||
|
output.set_destination(self.destination_address);
|
||||||
|
|
||||||
|
// Write the payload
|
||||||
|
output.set_payload(&payload);
|
||||||
|
|
||||||
|
// Return the packet
|
||||||
|
output.to_immutable().packet().to_vec()
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,6 @@
|
|||||||
pub mod icmp;
|
pub mod icmp;
|
||||||
pub mod icmpv6;
|
pub mod icmpv6;
|
||||||
|
pub mod ipv4;
|
||||||
|
pub mod ipv6;
|
||||||
pub mod tcp;
|
pub mod tcp;
|
||||||
pub mod udp;
|
pub mod udp;
|
||||||
|
@ -188,8 +188,9 @@ where
|
|||||||
let payload: Vec<u8> = self.payload.into();
|
let payload: Vec<u8> = self.payload.into();
|
||||||
|
|
||||||
// Allocate a mutable packet to write into
|
// Allocate a mutable packet to write into
|
||||||
let total_length =
|
let total_length = pnet_packet::tcp::MutableTcpPacket::minimum_packet_size()
|
||||||
pnet_packet::tcp::MutableTcpPacket::minimum_packet_size() + payload.len();
|
+ (self.options_length_words() as usize * 4)
|
||||||
|
+ payload.len();
|
||||||
let mut output =
|
let mut output =
|
||||||
pnet_packet::tcp::MutableTcpPacket::owned(vec![0u8; total_length]).unwrap();
|
pnet_packet::tcp::MutableTcpPacket::owned(vec![0u8; total_length]).unwrap();
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user