use std::net::IpAddr;
use std::time::{Duration, Instant};
use futures::stream::{self, StreamExt};
#[derive(Debug, Clone)]
pub struct PingOutcome {
pub host: IpAddr,
pub up: bool,
pub ttl: Option<u8>,
pub latency_ms: Option<u128>,
}
pub async fn ping_hosts(hosts: &[IpAddr], concurrency: usize) -> Vec<PingOutcome> {
let c = concurrency.max(1);
stream::iter(hosts.iter().copied())
.map(ping_one)
.buffer_unordered(c)
.collect()
.await
}
async fn ping_one(host: IpAddr) -> PingOutcome {
match tokio::task::spawn_blocking(move || raw_icmp_ping(host, Duration::from_secs(2))).await {
Ok(Some(outcome)) => outcome,
_ => system_ping_one(host).await,
}
}
fn raw_icmp_ping(host: IpAddr, timeout: Duration) -> Option<PingOutcome> {
match host {
IpAddr::V4(addr) => raw_icmp_ping_v4(addr, timeout),
IpAddr::V6(addr) => raw_icmp_ping_v6(addr, timeout),
}
}
fn raw_icmp_ping_v4(addr: std::net::Ipv4Addr, timeout: Duration) -> Option<PingOutcome> {
use pnet::packet::icmp::echo_request::MutableEchoRequestPacket;
use pnet::packet::icmp::IcmpTypes;
use pnet::packet::ip::IpNextHeaderProtocols;
use pnet::packet::Packet;
use pnet::transport::{
icmp_packet_iter, transport_channel, TransportChannelType, TransportProtocol,
};
let (mut tx, mut rx) = transport_channel(
4096,
TransportChannelType::Layer4(TransportProtocol::Ipv4(IpNextHeaderProtocols::Icmp)),
)
.ok()?;
let mut buf = vec![0u8; 64]; let mut pkt = MutableEchoRequestPacket::new(&mut buf)?;
pkt.set_icmp_type(IcmpTypes::EchoRequest);
pkt.set_icmp_code(pnet::packet::icmp::echo_request::IcmpCodes::NoCode);
pkt.set_identifier(std::process::id() as u16);
pkt.set_sequence_number(1);
let ck = {
let data = pkt.packet();
let mut sum = 0u32;
let mut i = 0;
while i + 1 < data.len() {
sum += u16::from_be_bytes([data[i], data[i + 1]]) as u32;
i += 2;
}
if i < data.len() {
sum += (data[i] as u32) << 8;
}
while sum >> 16 != 0 {
sum = (sum & 0xffff) + (sum >> 16);
}
!(sum as u16)
};
pkt.set_checksum(ck);
let start = Instant::now();
tx.send_to(pkt, IpAddr::V4(addr)).ok()?;
let mut iter = icmp_packet_iter(&mut rx);
let deadline = start + timeout;
loop {
let remaining = deadline.saturating_duration_since(Instant::now());
if remaining.is_zero() {
return Some(PingOutcome {
host: IpAddr::V4(addr),
up: false,
ttl: None,
latency_ms: None,
});
}
match iter.next_with_timeout(remaining) {
Ok(Some((pkt, src))) => {
use pnet::packet::icmp::IcmpPacket;
if let Some(icmp) = IcmpPacket::new(pkt.packet()) {
if icmp.get_icmp_type() == IcmpTypes::EchoReply {
if let IpAddr::V4(src_v4) = src {
if src_v4 == addr {
let elapsed = start.elapsed().as_millis();
return Some(PingOutcome {
host: IpAddr::V4(addr),
up: true,
ttl: None, latency_ms: Some(elapsed),
});
}
}
}
}
}
Ok(None) => {}
Err(_) => break,
}
}
Some(PingOutcome {
host: IpAddr::V4(addr),
up: false,
ttl: None,
latency_ms: None,
})
}
fn raw_icmp_ping_v6(addr: std::net::Ipv6Addr, timeout: Duration) -> Option<PingOutcome> {
use pnet::packet::icmpv6::Icmpv6Types;
use pnet::packet::icmpv6::MutableIcmpv6Packet;
use pnet::packet::ip::IpNextHeaderProtocols;
use pnet::transport::{
icmpv6_packet_iter, transport_channel, TransportChannelType, TransportProtocol,
};
let (mut tx, mut rx) = transport_channel(
4096,
TransportChannelType::Layer4(TransportProtocol::Ipv6(IpNextHeaderProtocols::Icmpv6)),
)
.ok()?;
let mut buf = vec![0u8; 64];
let mut pkt = MutableIcmpv6Packet::new(&mut buf)?;
pkt.set_icmpv6_type(Icmpv6Types::EchoRequest);
pkt.set_icmpv6_code(pnet::packet::icmpv6::Icmpv6Code::new(0));
pkt.set_checksum(0);
let start = Instant::now();
tx.send_to(pkt, IpAddr::V6(addr)).ok()?;
let mut iter = icmpv6_packet_iter(&mut rx);
let deadline = start + timeout;
loop {
let remaining = deadline.saturating_duration_since(Instant::now());
if remaining.is_zero() {
return Some(PingOutcome {
host: IpAddr::V6(addr),
up: false,
ttl: None,
latency_ms: None,
});
}
match iter.next_with_timeout(remaining) {
Ok(Some((pkt, src))) => {
if pkt.get_icmpv6_type() == Icmpv6Types::EchoReply {
if let IpAddr::V6(src_v6) = src {
if src_v6 == addr {
let elapsed = start.elapsed().as_millis();
return Some(PingOutcome {
host: IpAddr::V6(addr),
up: true,
ttl: None,
latency_ms: Some(elapsed),
});
}
}
}
}
Ok(None) => {}
Err(_) => break,
}
}
Some(PingOutcome {
host: IpAddr::V6(addr),
up: false,
ttl: None,
latency_ms: None,
})
}
async fn system_ping_one(host: IpAddr) -> PingOutcome {
use tokio::process::Command;
let start = Instant::now();
let (prog, args) = ping_cmd(host);
let mut cmd = Command::new(prog);
cmd.args(&args);
cmd.arg(host.to_string());
cmd.kill_on_drop(true);
match cmd.output().await {
Ok(out) => {
let ok = out.status.success();
let stdout = String::from_utf8_lossy(&out.stdout);
let ttl = parse_ttl(&stdout);
let latency_ms = parse_time_ms(&stdout).or_else(|| {
if ok {
Some(start.elapsed().as_millis())
} else {
None
}
});
PingOutcome {
host,
up: ok,
ttl,
latency_ms,
}
}
Err(_) => PingOutcome {
host,
up: false,
ttl: None,
latency_ms: None,
},
}
}
fn ping_cmd(host: IpAddr) -> (&'static str, Vec<&'static str>) {
#[cfg(windows)]
{
match host {
IpAddr::V4(_) => ("ping", vec!["-n", "1", "-w", "1000"]),
IpAddr::V6(_) => ("ping", vec!["-6", "-n", "1", "-w", "1000"]),
}
}
#[cfg(target_os = "macos")]
{
match host {
IpAddr::V4(_) => ("ping", vec!["-c", "1", "-W", "1000"]),
IpAddr::V6(_) => ("ping6", vec!["-c", "1", "-W", "1000"]),
}
}
#[cfg(all(unix, not(target_os = "macos")))]
{
match host {
IpAddr::V4(_) => ("ping", vec!["-c", "1", "-W", "1"]),
IpAddr::V6(_) => ("ping6", vec!["-c", "1", "-W", "1"]),
}
}
}
fn parse_ttl(s: &str) -> Option<u8> {
for part in s.split_whitespace() {
let p = part.to_ascii_lowercase();
if let Some(rest) = p.strip_prefix("ttl=") {
return rest.trim_end_matches(',').parse().ok();
}
if let Some(rest) = p.strip_prefix("ttl") {
let rest = rest.trim_start_matches('=');
return rest.trim_end_matches(',').parse().ok();
}
}
None
}
fn parse_time_ms(s: &str) -> Option<u128> {
for line in s.lines() {
let l = line.to_ascii_lowercase();
if let Some(idx) = l.find("time=") {
let tail = &line[idx + 5..];
let num = tail
.split(|c: char| !c.is_ascii_digit() && c != '.')
.next()
.unwrap_or("");
if let Ok(ms) = num.parse::<f64>() {
return Some(ms as u128);
}
}
}
None
}
#[cfg(test)]
mod tests {
use super::{parse_time_ms, parse_ttl, ping_cmd};
use std::net::{IpAddr, Ipv4Addr, Ipv6Addr};
#[test]
fn macos_ping_ttl_line_extracts_64() {
let s = "64 bytes from 8.8.8.8: icmp_seq=0 ttl=128 time=10.523 ms";
assert_eq!(parse_ttl(s), Some(128));
}
#[test]
fn linux_iputils_ttl_line_extracts() {
let s = "64 bytes from 1.1.1.1: icmp_seq=1 ttl=57 time=12.3 ms";
assert_eq!(parse_ttl(s), Some(57));
}
#[test]
fn windows_ttl_uppercase_extracts() {
let s = "Reply from 8.8.8.8: bytes=32 time=15ms TTL=128";
assert_eq!(parse_ttl(s), Some(128));
}
#[test]
fn ttl_with_trailing_comma_strips_comma() {
let s = "ttl=64,";
assert_eq!(parse_ttl(s), Some(64));
}
#[test]
fn ttl_max_u8_is_255() {
let s = "ttl=255";
assert_eq!(parse_ttl(s), Some(255));
}
#[test]
fn ttl_overflow_returns_none() {
let s = "ttl=256";
assert_eq!(parse_ttl(s), None);
}
#[test]
fn no_ttl_in_string_returns_none() {
assert_eq!(parse_ttl("nothing relevant here"), None);
assert_eq!(parse_ttl(""), None);
}
#[test]
fn ttl_mixed_case_matches() {
assert_eq!(parse_ttl("TtL=42"), Some(42));
assert_eq!(parse_ttl("Ttl=42"), Some(42));
}
#[test]
fn macos_time_line_extracts_milliseconds() {
let s = "64 bytes from 8.8.8.8: icmp_seq=0 ttl=128 time=10.523 ms";
assert_eq!(parse_time_ms(s), Some(10));
}
#[test]
fn windows_time_no_decimal_extracts_integer_ms() {
let s = "Reply from 8.8.8.8: bytes=32 time=15ms TTL=128";
assert_eq!(parse_time_ms(s), Some(15));
}
#[test]
fn time_less_than_one_ms_truncates_to_zero() {
let s = "time=0.5 ms";
assert_eq!(parse_time_ms(s), Some(0));
}
#[test]
fn time_searched_in_each_line() {
let s = "PING google.com\n64 bytes from 8.8.8.8: time=5.5 ms\n";
assert_eq!(parse_time_ms(s), Some(5));
}
#[test]
fn no_time_returns_none() {
assert_eq!(parse_time_ms("nothing here"), None);
assert_eq!(parse_time_ms(""), None);
}
#[test]
fn invalid_time_value_returns_none() {
let s = "time=abc";
assert_eq!(parse_time_ms(s), None);
}
#[test]
fn ping_cmd_v4_uses_ping_binary() {
let (prog, _args) = ping_cmd(IpAddr::V4(Ipv4Addr::new(1, 1, 1, 1)));
assert_eq!(prog, "ping");
}
#[test]
#[cfg(target_os = "macos")]
fn ping_cmd_v6_uses_ping6_on_macos() {
let (prog, _args) = ping_cmd(IpAddr::V6(Ipv6Addr::LOCALHOST));
assert_eq!(prog, "ping6");
}
#[test]
#[cfg(all(unix, not(target_os = "macos")))]
fn ping_cmd_v6_uses_ping6_on_linux() {
let (prog, _args) = ping_cmd(IpAddr::V6(Ipv6Addr::LOCALHOST));
assert_eq!(prog, "ping6");
}
#[test]
#[cfg(unix)]
fn ping_cmd_count_one_arg_present() {
let (_prog, args) = ping_cmd(IpAddr::V4(Ipv4Addr::new(1, 1, 1, 1)));
assert!(args.contains(&"-c"));
assert!(args.contains(&"1"));
}
#[test]
fn ttl_zero_is_valid() {
assert_eq!(parse_ttl("ttl=0"), Some(0));
}
#[test]
fn ttl_non_numeric_suffix_returns_none() {
assert_eq!(parse_ttl("ttl=abc"), None);
}
#[test]
fn parse_time_ms_integer_without_decimal() {
assert_eq!(parse_time_ms("time=42 ms"), Some(42));
}
#[test]
fn parse_time_ms_picks_first_match_in_multiline() {
let s = "time=9 ms\ntime=99 ms\n";
assert_eq!(parse_time_ms(s), Some(9));
}
#[test]
fn parse_ttl_before_time_in_same_line() {
let s = "ttl=32 time=7.2 ms";
assert_eq!(parse_ttl(s), Some(32));
assert_eq!(parse_time_ms(s), Some(7));
}
#[test]
#[cfg(unix)]
fn ping_cmd_v6_uses_ping6_binary() {
let host = IpAddr::V6(Ipv6Addr::new(0x2001, 0xdb8, 0, 0, 0, 0, 0, 1));
let (prog, _args) = ping_cmd(host);
assert_eq!(prog, "ping6");
}
#[test]
fn parse_ttl_on_second_line_only() {
let s = "header line\n64 bytes from x: ttl=33 time=1 ms\n";
assert_eq!(parse_ttl(s), Some(33));
}
#[test]
fn parse_ttl_ttl_without_equals_sign() {
let s = "Reply from x: bytes=32 TTL128 time=1ms";
assert_eq!(parse_ttl(s), Some(128));
}
#[test]
fn parse_time_ms_large_value_truncates() {
assert_eq!(parse_time_ms("time=9999.9 ms"), Some(9999));
}
#[test]
fn parse_time_ms_no_unit_still_parses_number() {
assert_eq!(parse_time_ms("time=42"), Some(42));
}
#[test]
fn parse_ttl_multiple_ttl_tokens_first_wins() {
let s = "ttl=10 ttl=20";
assert_eq!(parse_ttl(s), Some(10));
}
#[test]
fn parse_time_ms_only_positive_integers() {
assert_eq!(parse_time_ms("time=5 ms"), Some(5));
assert!(parse_time_ms("time=0 ms") == Some(0));
}
#[test]
fn parse_ttl_with_spaces_around_token() {
assert_eq!(parse_ttl(" ttl=64 "), Some(64));
}
#[test]
fn parse_time_ms_fractional_submillisecond_truncates() {
assert_eq!(parse_time_ms("time=0.999 ms"), Some(0));
}
#[test]
fn parse_ttl_in_mixed_case_line() {
assert_eq!(parse_ttl("Reply TTL=48 from host"), Some(48));
}
#[test]
fn parse_time_ms_windows_style_no_space_before_ms() {
assert_eq!(parse_time_ms("time=15ms"), Some(15));
}
#[test]
fn parse_ttl_trailing_comma_stripped() {
assert_eq!(parse_ttl("ttl=48,"), Some(48));
}
#[test]
fn parse_ttl_uppercase_ttl_prefix() {
assert_eq!(parse_ttl("TTL=32"), Some(32));
}
#[test]
fn parse_time_ms_zero_ms() {
assert_eq!(parse_time_ms("time=0.0 ms"), Some(0));
}
#[test]
fn parse_ttl_no_match_returns_none() {
assert_eq!(parse_ttl("bytes=32 from host"), None);
}
#[test]
fn parse_time_ms_no_match_returns_none() {
assert_eq!(parse_time_ms("no timing here"), None);
}
#[test]
fn parse_ttl_equals_sign_required_for_lowercase() {
assert_eq!(parse_ttl("ttl 64"), None);
}
#[test]
fn parse_time_ms_negative_not_matched() {
assert_eq!(parse_time_ms("time=-5 ms"), None);
}
#[test]
fn parse_ttl_max_u8() {
assert_eq!(parse_ttl("ttl=255"), Some(255));
}
#[test]
fn parse_time_ms_decimal_only_fraction() {
assert_eq!(parse_time_ms("time=.5 ms"), Some(0));
}
#[test]
fn parse_ttl_in_parentheses_not_matched() {
assert_eq!(parse_ttl("(ttl=47)"), None);
}
#[test]
fn parse_time_ms_tab_separated() {
assert_eq!(parse_time_ms("time=12\tms"), Some(12));
}
#[test]
fn parse_ttl_mixed_with_bytes_from() {
let s = "64 bytes from 127.0.0.1: icmp_seq=1 ttl=64 time=0.1 ms";
assert_eq!(parse_ttl(s), Some(64));
assert_eq!(parse_time_ms(s), Some(0));
}
#[test]
fn parse_time_ms_multiple_decimals_invalid() {
assert_eq!(parse_time_ms("time=1.2.3 ms"), None);
}
#[test]
fn parse_ttl_leading_zero() {
assert_eq!(parse_ttl("ttl=064"), Some(64));
}
#[test]
fn ping_cmd_v4_returns_ping_binary() {
let (prog, args) = ping_cmd(IpAddr::V4(Ipv4Addr::new(203, 0, 113, 1)));
assert_eq!(prog, "ping");
assert!(args.contains(&"-c"));
}
#[test]
fn parse_time_ms_uppercase_time_prefix() {
assert_eq!(parse_time_ms("TIME=7 ms"), Some(7));
}
#[test]
fn parse_ttl_after_time_still_found() {
let s = "time=1 ms ttl=50";
assert_eq!(parse_ttl(s), Some(50));
}
}