Log common errors instead of panic
This commit is contained in:
parent
d5dfdd012b
commit
555d1a1646
@ -9,7 +9,7 @@ use crate::{
|
|||||||
use self::{
|
use self::{
|
||||||
error::Nat64Error,
|
error::Nat64Error,
|
||||||
table::Nat64Table,
|
table::Nat64Table,
|
||||||
utils::{embed_address, extract_address},
|
utils::{embed_address, extract_address, unwrap_log},
|
||||||
};
|
};
|
||||||
use ipnet::{Ipv4Net, Ipv6Net};
|
use ipnet::{Ipv4Net, Ipv6Net};
|
||||||
use protomask_tun::TunDevice;
|
use protomask_tun::TunDevice;
|
||||||
@ -99,11 +99,14 @@ impl Nat64 {
|
|||||||
|
|
||||||
// Spawn a task to process the packet
|
// Spawn a task to process the packet
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let output =
|
if let Some(output) = unwrap_log(translate_ipv4_to_ipv6(
|
||||||
translate_ipv4_to_ipv6(packet, new_source, new_destination)
|
packet,
|
||||||
.unwrap();
|
new_source,
|
||||||
tx.send(output.into()).await.unwrap();
|
new_destination,
|
||||||
PACKET_COUNTER.with_label_values(&["ipv6", "sent"]).inc();
|
)) {
|
||||||
|
tx.send(output.into()).await.unwrap();
|
||||||
|
PACKET_COUNTER.with_label_values(&["ipv6", "sent"]).inc();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
6 => {
|
6 => {
|
||||||
@ -145,11 +148,14 @@ impl Nat64 {
|
|||||||
|
|
||||||
// Spawn a task to process the packet
|
// Spawn a task to process the packet
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let output =
|
if let Some(output) = unwrap_log(translate_ipv6_to_ipv4(
|
||||||
translate_ipv6_to_ipv4(packet, new_source, new_destination)
|
packet,
|
||||||
.unwrap();
|
new_source,
|
||||||
tx.send(output.into()).await.unwrap();
|
new_destination,
|
||||||
PACKET_COUNTER.with_label_values(&["ipv4", "sent"]).inc();
|
)) {
|
||||||
|
tx.send(output.into()).await.unwrap();
|
||||||
|
PACKET_COUNTER.with_label_values(&["ipv4", "sent"]).inc();
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
n => {
|
n => {
|
||||||
|
@ -2,6 +2,8 @@ use std::net::{Ipv4Addr, Ipv6Addr};
|
|||||||
|
|
||||||
use ipnet::Ipv6Net;
|
use ipnet::Ipv6Net;
|
||||||
|
|
||||||
|
use crate::packet::error::PacketError;
|
||||||
|
|
||||||
/// Embed an IPv4 address in an IPv6 prefix
|
/// Embed an IPv4 address in an IPv6 prefix
|
||||||
pub fn embed_address(ipv4_address: Ipv4Addr, ipv6_prefix: Ipv6Net) -> Ipv6Addr {
|
pub fn embed_address(ipv4_address: Ipv4Addr, ipv6_prefix: Ipv6Net) -> Ipv6Addr {
|
||||||
let v4_octets = ipv4_address.octets();
|
let v4_octets = ipv4_address.octets();
|
||||||
@ -23,3 +25,33 @@ pub fn extract_address(ipv6_address: Ipv6Addr) -> Ipv4Addr {
|
|||||||
let octets = ipv6_address.octets();
|
let octets = ipv6_address.octets();
|
||||||
Ipv4Addr::new(octets[12], octets[13], octets[14], octets[15])
|
Ipv4Addr::new(octets[12], octets[13], octets[14], octets[15])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Logs errors instead of crashing out of them
|
||||||
|
pub fn unwrap_log<T>(result: Result<T, PacketError>) -> Option<T> {
|
||||||
|
match result {
|
||||||
|
Ok(value) => Some(value),
|
||||||
|
Err(err) => match err {
|
||||||
|
PacketError::MismatchedAddressFamily(addr_a, addr_b) => {
|
||||||
|
log::error!(
|
||||||
|
"Mismatched address family between {} and {}",
|
||||||
|
addr_a,
|
||||||
|
addr_b
|
||||||
|
);
|
||||||
|
None
|
||||||
|
}
|
||||||
|
PacketError::TooShort(len, data) => {
|
||||||
|
log::warn!("Received packet that's too short to parse. Length {}", len);
|
||||||
|
log::debug!("Short packet: {:?}", data);
|
||||||
|
None
|
||||||
|
}
|
||||||
|
PacketError::UnsupportedIcmpType(icmp_type) => {
|
||||||
|
log::warn!("Unsupported ICMP type {}", icmp_type);
|
||||||
|
None
|
||||||
|
}
|
||||||
|
PacketError::UnsupportedIcmpv6Type(icmp_type) => {
|
||||||
|
log::warn!("Unsupported ICMPv6 type {}", icmp_type);
|
||||||
|
None
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -5,7 +5,7 @@ pub enum PacketError {
|
|||||||
#[error("Mismatched source and destination address family: source={0:?}, destination={1:?}")]
|
#[error("Mismatched source and destination address family: source={0:?}, destination={1:?}")]
|
||||||
MismatchedAddressFamily(IpAddr, IpAddr),
|
MismatchedAddressFamily(IpAddr, IpAddr),
|
||||||
#[error("Packet too short: {0}")]
|
#[error("Packet too short: {0}")]
|
||||||
TooShort(usize),
|
TooShort(usize, Vec<u8>),
|
||||||
#[error("Unsupported ICMP type: {0}")]
|
#[error("Unsupported ICMP type: {0}")]
|
||||||
UnsupportedIcmpType(u8),
|
UnsupportedIcmpType(u8),
|
||||||
#[error("Unsupported ICMPv6 type: {0}")]
|
#[error("Unsupported ICMPv6 type: {0}")]
|
||||||
|
@ -31,8 +31,8 @@ where
|
|||||||
|
|
||||||
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
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)
|
||||||
pnet_packet::icmp::IcmpPacket::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
.ok_or(PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@ -43,7 +43,6 @@ 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>>,
|
||||||
|
@ -50,7 +50,7 @@ where
|
|||||||
) -> Result<Self, PacketError> {
|
) -> Result<Self, PacketError> {
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let packet = pnet_packet::icmpv6::Icmpv6Packet::new(bytes)
|
let packet = pnet_packet::icmpv6::Icmpv6Packet::new(bytes)
|
||||||
.ok_or(PacketError::TooShort(bytes.len()))?;
|
.ok_or(PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@ -72,7 +72,7 @@ impl Icmpv6Packet<RawBytes> {
|
|||||||
) -> Result<Self, PacketError> {
|
) -> Result<Self, PacketError> {
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let packet = pnet_packet::icmpv6::Icmpv6Packet::new(bytes)
|
let packet = pnet_packet::icmpv6::Icmpv6Packet::new(bytes)
|
||||||
.ok_or(PacketError::TooShort(bytes.len()))?;
|
.ok_or(PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
@ -71,7 +71,7 @@ where
|
|||||||
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let packet =
|
let packet =
|
||||||
pnet_packet::ipv4::Ipv4Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
pnet_packet::ipv4::Ipv4Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
@ -47,7 +47,7 @@ where
|
|||||||
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
fn try_from(bytes: Vec<u8>) -> Result<Self, Self::Error> {
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let packet =
|
let packet =
|
||||||
pnet_packet::ipv6::Ipv6Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len()))?;
|
pnet_packet::ipv6::Ipv6Packet::new(&bytes).ok_or(PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Return the packet
|
// Return the packet
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
@ -131,7 +131,7 @@ where
|
|||||||
|
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let parsed = pnet_packet::tcp::TcpPacket::new(bytes)
|
let parsed = pnet_packet::tcp::TcpPacket::new(bytes)
|
||||||
.ok_or_else(|| PacketError::TooShort(bytes.len()))?;
|
.ok_or_else(|| PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Build the struct
|
// Build the struct
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@ -165,7 +165,7 @@ impl TcpPacket<RawBytes> {
|
|||||||
|
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let parsed = pnet_packet::tcp::TcpPacket::new(bytes)
|
let parsed = pnet_packet::tcp::TcpPacket::new(bytes)
|
||||||
.ok_or_else(|| PacketError::TooShort(bytes.len()))?;
|
.ok_or_else(|| PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Build the struct
|
// Build the struct
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
@ -102,7 +102,7 @@ where
|
|||||||
|
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let parsed = pnet_packet::udp::UdpPacket::new(bytes)
|
let parsed = pnet_packet::udp::UdpPacket::new(bytes)
|
||||||
.ok_or_else(|| PacketError::TooShort(bytes.len()))?;
|
.ok_or_else(|| PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Build the struct
|
// Build the struct
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
@ -130,7 +130,7 @@ impl UdpPacket<RawBytes> {
|
|||||||
|
|
||||||
// Parse the packet
|
// Parse the packet
|
||||||
let parsed = pnet_packet::udp::UdpPacket::new(bytes)
|
let parsed = pnet_packet::udp::UdpPacket::new(bytes)
|
||||||
.ok_or_else(|| PacketError::TooShort(bytes.len()))?;
|
.ok_or_else(|| PacketError::TooShort(bytes.len(), bytes.to_vec()))?;
|
||||||
|
|
||||||
// Build the struct
|
// Build the struct
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user