mirror of
https://github.com/Astatin3/rust-scan.git
synced 2026-06-08 16:18:07 -06:00
Create the bandwidth destroyer
This commit is contained in:
+10
@@ -0,0 +1,10 @@
|
|||||||
|
[package]
|
||||||
|
name = "untitled"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2024"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
loadingbar = "1.0.1"
|
||||||
|
pnet = "0.35.0"
|
||||||
|
rand = "0.9.0"
|
||||||
|
rocksdb = "0.23.0"
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
pub mod online_scan;
|
||||||
|
pub mod parse_ip_range;
|
||||||
+28
@@ -0,0 +1,28 @@
|
|||||||
|
pub mod online_scan;
|
||||||
|
pub mod parse_ip_range;
|
||||||
|
|
||||||
|
use std::env;
|
||||||
|
|
||||||
|
use online_scan::ping_scanner;
|
||||||
|
use parse_ip_range::parse_ip_targets;
|
||||||
|
|
||||||
|
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let args: Vec<String> = env::args().collect();
|
||||||
|
|
||||||
|
// Set default targets or use command line input
|
||||||
|
let targets = if args.len() > 1 {
|
||||||
|
args[1].clone()
|
||||||
|
} else {
|
||||||
|
"".to_string()
|
||||||
|
};
|
||||||
|
|
||||||
|
// Parse the targets into IP addresses
|
||||||
|
let hosts = parse_ip_targets(&targets)?;
|
||||||
|
|
||||||
|
let length = hosts.len();
|
||||||
|
|
||||||
|
let results = ping_scanner::ping_scan(hosts).unwrap();
|
||||||
|
|
||||||
|
println!("Finished! {} Scanned, {} Up", length, results.len());
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
pub mod online_scan;
|
||||||
|
pub mod ping_scanner;
|
||||||
|
|
||||||
|
pub use online_scan::PingResult;
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
use std::{net::IpAddr, time::Duration};
|
||||||
|
|
||||||
|
// Structure to hold ping results
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct PingResult {
|
||||||
|
pub host: IpAddr,
|
||||||
|
pub is_up: bool,
|
||||||
|
pub response_time: Option<Duration>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PingResult {
|
||||||
|
pub fn create(addr: IpAddr) -> Self {
|
||||||
|
Self {
|
||||||
|
host: addr,
|
||||||
|
is_up: false,
|
||||||
|
response_time: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,173 @@
|
|||||||
|
use pnet::packet::ip::IpNextHeaderProtocols;
|
||||||
|
use pnet::packet::ipv4::Ipv4OptionNumbers::TR;
|
||||||
|
use pnet::packet::{
|
||||||
|
Packet,
|
||||||
|
icmp::{IcmpTypes, echo_request::MutableEchoRequestPacket},
|
||||||
|
};
|
||||||
|
use pnet::transport::{
|
||||||
|
TransportChannelType, TransportProtocol, icmp_packet_iter, transport_channel,
|
||||||
|
};
|
||||||
|
use pnet::util::checksum;
|
||||||
|
use std::cmp::max;
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::net::IpAddr;
|
||||||
|
use std::sync::atomic::{AtomicBool, Ordering};
|
||||||
|
use std::sync::{Arc, Mutex};
|
||||||
|
use std::thread;
|
||||||
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
|
static TIMEOUT: Duration = Duration::from_secs(3);
|
||||||
|
// static MAX_PINGS_PER_SECOND: u64 = 10000;
|
||||||
|
static SEND_DELAY_NANOS: Duration = Duration::from_nanos(500);
|
||||||
|
|
||||||
|
use crate::online_scan::PingResult;
|
||||||
|
|
||||||
|
pub fn ping_scan(hosts: Vec<IpAddr>) -> Result<Vec<PingResult>, Box<dyn std::error::Error>> {
|
||||||
|
let results = Arc::new(Mutex::new(Vec::<PingResult>::new()));
|
||||||
|
|
||||||
|
// Create a receiver channel for ICMP packets
|
||||||
|
let (_, mut rx) = transport_channel(
|
||||||
|
1024,
|
||||||
|
TransportChannelType::Layer4(TransportProtocol::Ipv4(IpNextHeaderProtocols::Icmp)),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
// Create a map to store host identifiers
|
||||||
|
let requests: Arc<Mutex<HashMap<u16, IpAddr>>> = Arc::new(Mutex::new(HashMap::new()));
|
||||||
|
|
||||||
|
let finished_sending_time = Arc::new(AtomicBool::new(false));
|
||||||
|
|
||||||
|
// Set up the receiver thread
|
||||||
|
// let recv_identifiers = Arc::clone(&identifiers);
|
||||||
|
let recv_results = Arc::clone(&results);
|
||||||
|
let recv_requests = Arc::clone(&requests);
|
||||||
|
let recv_finished_sending_time = Arc::clone(&finished_sending_time);
|
||||||
|
let receiver_handle = thread::spawn(move || {
|
||||||
|
let mut iter = icmp_packet_iter(&mut rx);
|
||||||
|
let start_time = Instant::now();
|
||||||
|
let mut finish_sending_time: Option<Instant> = None;
|
||||||
|
|
||||||
|
// Keep receiving until timeout or all hosts are accounted for
|
||||||
|
loop {
|
||||||
|
// Stop reciving loop if timeout is reached
|
||||||
|
// let time = finished_sending_time;
|
||||||
|
if finish_sending_time.is_some() && finish_sending_time.unwrap().elapsed() >= TIMEOUT {
|
||||||
|
break;
|
||||||
|
} else if finish_sending_time.is_none()
|
||||||
|
&& recv_finished_sending_time.load(Ordering::Relaxed)
|
||||||
|
{
|
||||||
|
finish_sending_time = Some(Instant::now());
|
||||||
|
println!("Waiting {} seconds for timeout...", TIMEOUT.as_secs())
|
||||||
|
}
|
||||||
|
// if time.is_some() {
|
||||||
|
// println!("{}", time.unwrap().elapsed().as_millis())
|
||||||
|
// }
|
||||||
|
// if time.is_some() && time.unwrap().elapsed() >= TIMEOUT {
|
||||||
|
// break;
|
||||||
|
// };
|
||||||
|
|
||||||
|
match iter.next_with_timeout(Duration::from_millis(1)) {
|
||||||
|
Ok(Some((packet, _))) => {
|
||||||
|
if packet.get_icmp_type() == IcmpTypes::EchoReply {
|
||||||
|
let payload = packet.payload();
|
||||||
|
let id = ((payload[2] as u16) << 8) + (payload[3] as u16);
|
||||||
|
|
||||||
|
let host_option = {
|
||||||
|
let ids = recv_requests.lock().unwrap();
|
||||||
|
ids.get(&id).cloned()
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(host) = host_option {
|
||||||
|
let mut results = recv_results.lock().unwrap();
|
||||||
|
let response_time = start_time.elapsed();
|
||||||
|
results.push(PingResult {
|
||||||
|
host,
|
||||||
|
is_up: true,
|
||||||
|
response_time: Some(response_time),
|
||||||
|
});
|
||||||
|
|
||||||
|
// println!("Up! {0} {1}ms", host, response_time.as_millis());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(None) => { /* Timeout, continue */ }
|
||||||
|
Err(_) => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Spawn sender threads
|
||||||
|
|
||||||
|
let sender_requests = Arc::clone(&requests);
|
||||||
|
let sender_results = Arc::clone(&results);
|
||||||
|
let sender_finished_sending_time = Arc::clone(&finished_sending_time);
|
||||||
|
let sender_handle = thread::spawn(move || {
|
||||||
|
// let mut last_send_time = Instant::now();
|
||||||
|
for (i, host) in hosts.iter().enumerate() {
|
||||||
|
let host_clone = host.clone();
|
||||||
|
|
||||||
|
// Use the index as a unique identifier for each host
|
||||||
|
let identifier: u16 = i as u16;
|
||||||
|
|
||||||
|
// Store the host-identifier mapping
|
||||||
|
{
|
||||||
|
let mut ids = sender_requests.lock().unwrap();
|
||||||
|
ids.insert(identifier, host_clone);
|
||||||
|
}
|
||||||
|
|
||||||
|
let response = send_ping(host_clone, identifier);
|
||||||
|
|
||||||
|
match response {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(_) => {
|
||||||
|
let mut results = sender_results.lock().unwrap();
|
||||||
|
results.push(PingResult {
|
||||||
|
host: host_clone,
|
||||||
|
is_up: false,
|
||||||
|
response_time: None,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// let now = Instant::now();
|
||||||
|
// let delay = MAX_RATE_NANOS - last_send_time.duration_since(now).as_nanos() as u64;
|
||||||
|
// last_send_time = now;
|
||||||
|
thread::sleep(SEND_DELAY_NANOS);
|
||||||
|
}
|
||||||
|
println!("Finished Sending!");
|
||||||
|
|
||||||
|
sender_finished_sending_time.swap(true, Ordering::Relaxed);
|
||||||
|
});
|
||||||
|
// Wait for all sender threads to complete
|
||||||
|
// for handle in sender_handles {a
|
||||||
|
// handle.join().unwrap();
|
||||||
|
// }
|
||||||
|
sender_handle.join().unwrap();
|
||||||
|
receiver_handle.join().unwrap();
|
||||||
|
|
||||||
|
let results = Arc::try_unwrap(results).unwrap().into_inner().unwrap();
|
||||||
|
Ok(results)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn send_ping(target: IpAddr, identifier: u16) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
// Create a transport channel
|
||||||
|
let (mut tx, _) = transport_channel(
|
||||||
|
64,
|
||||||
|
TransportChannelType::Layer4(TransportProtocol::Ipv4(IpNextHeaderProtocols::Icmp)),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
// Create an ICMP packet
|
||||||
|
let mut vec = vec![0; 8];
|
||||||
|
let mut echo_packet = MutableEchoRequestPacket::new(&mut vec[..]).unwrap();
|
||||||
|
|
||||||
|
// Fill in the ICMP packet details
|
||||||
|
echo_packet.set_icmp_type(IcmpTypes::EchoRequest);
|
||||||
|
echo_packet.set_sequence_number(identifier);
|
||||||
|
|
||||||
|
let checksum = checksum(echo_packet.packet(), 1);
|
||||||
|
echo_packet.set_checksum(checksum);
|
||||||
|
|
||||||
|
// Send the ICMP packet
|
||||||
|
tx.send_to(echo_packet, target)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -0,0 +1,116 @@
|
|||||||
|
use std::{
|
||||||
|
net::{IpAddr, Ipv4Addr},
|
||||||
|
str::FromStr,
|
||||||
|
};
|
||||||
|
|
||||||
|
// static MAX_HOSTS: u32 = 1024;
|
||||||
|
|
||||||
|
/// Parse a comma-separated list of IP targets
|
||||||
|
/// Each target can be:
|
||||||
|
/// - Single IP: 192.168.1.1
|
||||||
|
/// - IP range: 192.168.1.1-192.168.1.10
|
||||||
|
/// - CIDR notation: 192.168.1.0/24
|
||||||
|
pub fn parse_ip_targets(targets: &str) -> Result<Vec<IpAddr>, Box<dyn std::error::Error>> {
|
||||||
|
let mut ips = Vec::new();
|
||||||
|
|
||||||
|
// Split the input by commas
|
||||||
|
for target in targets.split(',') {
|
||||||
|
let target = target.trim();
|
||||||
|
|
||||||
|
if target.contains('/') {
|
||||||
|
// CIDR notation
|
||||||
|
parse_cidr(target, &mut ips)?;
|
||||||
|
} else if target.contains('-') {
|
||||||
|
// IP range
|
||||||
|
parse_ip_range(target, &mut ips)?;
|
||||||
|
} else {
|
||||||
|
// Single IP
|
||||||
|
let ip = IpAddr::from_str(target)?;
|
||||||
|
ips.push(ip);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(ips)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse CIDR notation (e.g., 192.168.1.0/24)
|
||||||
|
fn parse_cidr(cidr: &str, ips: &mut Vec<IpAddr>) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let parts: Vec<&str> = cidr.split('/').collect();
|
||||||
|
if parts.len() != 2 {
|
||||||
|
return Err("Invalid CIDR format".into());
|
||||||
|
}
|
||||||
|
|
||||||
|
let base_ip = Ipv4Addr::from_str(parts[0])?;
|
||||||
|
let prefix_len: u8 = parts[1].parse()?;
|
||||||
|
|
||||||
|
if prefix_len > 32 {
|
||||||
|
return Err("Invalid CIDR prefix length".into());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the number of hosts in this CIDR
|
||||||
|
let hosts = if prefix_len == 32 {
|
||||||
|
1 // Just one host for /32
|
||||||
|
} else {
|
||||||
|
1u32 << (32 - prefix_len) // 2^(32-prefix_len)
|
||||||
|
};
|
||||||
|
|
||||||
|
// If the CIDR is too large, warn and limit
|
||||||
|
// if hosts > MAX_HOSTS {
|
||||||
|
// println!(
|
||||||
|
// "Warning: CIDR {} contains {} hosts. Limiting to first 1024.",
|
||||||
|
// cidr, hosts
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
|
// Convert base IP to u32 for easier manipulation
|
||||||
|
let base_ip_u32 = u32::from(base_ip);
|
||||||
|
|
||||||
|
// Generate all IPs in the CIDR block (up to limit)
|
||||||
|
// let limit = std::cmp::min(hosts, MAX_HOSTS);
|
||||||
|
for i in 0..hosts {
|
||||||
|
let ip_u32 = base_ip_u32 & (0xFFFFFFFF << (32 - prefix_len)) | i;
|
||||||
|
let ip = Ipv4Addr::from(ip_u32);
|
||||||
|
ips.push(IpAddr::V4(ip));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Parse IP range (e.g., 192.168.1.1-192.168.1.10)
|
||||||
|
fn parse_ip_range(range: &str, ips: &mut Vec<IpAddr>) -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let parts: Vec<&str> = range.split('-').collect();
|
||||||
|
if parts.len() != 2 {
|
||||||
|
return Err("Invalid IP range format".into());
|
||||||
|
}
|
||||||
|
|
||||||
|
let start_ip = Ipv4Addr::from_str(parts[0])?;
|
||||||
|
let end_ip = Ipv4Addr::from_str(parts[1])?;
|
||||||
|
|
||||||
|
let start_u32 = u32::from(start_ip);
|
||||||
|
let end_u32 = u32::from(end_ip);
|
||||||
|
|
||||||
|
if start_u32 > end_u32 {
|
||||||
|
return Err("Invalid IP range: start IP is greater than end IP".into());
|
||||||
|
}
|
||||||
|
|
||||||
|
let range_size = end_u32 - start_u32 + 1;
|
||||||
|
|
||||||
|
// // If the range is too large, warn and limit
|
||||||
|
// if range_size > MAX_HOSTS {
|
||||||
|
// println!(
|
||||||
|
// "Warning: IP range {}-{} contains {} hosts. Limiting to first 1024.",
|
||||||
|
// start_ip, end_ip, range_size
|
||||||
|
// );
|
||||||
|
// }
|
||||||
|
|
||||||
|
// let limit = std::cmp::min(range_size, MAX_HOSTS);
|
||||||
|
|
||||||
|
// Generate all IPs in the range (up to limit)
|
||||||
|
for i in 0..range_size {
|
||||||
|
let ip_u32 = start_u32 + i;
|
||||||
|
let ip = Ipv4Addr::from(ip_u32);
|
||||||
|
ips.push(IpAddr::V4(ip));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
pub mod tcp_scan;
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
|
||||||
Reference in New Issue
Block a user