contego/src/bin/client.rs

151 lines
5.0 KiB
Rust
Raw Normal View History

use clap::Parser;
2022-07-10 21:16:50 +02:00
use std::time::Duration;
use tokio::{
fs::File,
io::{AsyncBufReadExt, AsyncReadExt, AsyncWriteExt, BufReader, BufWriter},
net::TcpStream,
time::sleep,
};
// TODO: Remove panics/unwraps & add proper error handling
#[derive(Debug, Parser)]
#[clap(author, about, version)]
struct Args {
#[clap(short = 't', long, value_parser)]
target: String,
#[clap(default_value = "./output/", short = 'f', long, value_parser)]
fileroot: String,
}
2022-07-10 21:16:50 +02:00
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
let args = Args::parse();
let addr = args.target;
let fileroot = args.fileroot;
let mut stream = TcpStream::connect(addr.clone()).await?;
2022-07-10 21:16:50 +02:00
println!("[+] Connecting to {}", addr);
let (reader, writer) = stream.split();
let mut reader = BufReader::new(reader);
let mut writer = BufWriter::new(writer);
let mut buf = Vec::new();
loop {
let bytes_read = reader.read_buf(&mut buf).await.unwrap();
if bytes_read == 0 {
println!("[-] No more bytes received, closing connection");
2022-07-10 21:16:50 +02:00
break;
}
// Receive buffersize
let buffersize = String::from_utf8(buf.clone())
.unwrap()
.parse::<usize>()
.unwrap();
println!("[+] Selected buffersize: {}", buffersize);
buf.clear();
// ACK buffersize
writer.write_all(b"ACK").await.unwrap();
writer.flush().await.unwrap();
2022-07-10 21:16:50 +02:00
// Receive file amount
let _bytes_read = reader.read_buf(&mut buf).await.unwrap();
2022-07-10 21:16:50 +02:00
let file_amount = String::from_utf8(buf.clone())
.unwrap()
.parse::<usize>()
.unwrap();
println!("[+] Total of {} files available", file_amount);
buf.clear();
// ACK file amount
writer.write_all(b"ACK").await.unwrap();
writer.flush().await.unwrap();
// Receive file metadata
println!("[+] Receiving file metadata");
let mut metadata = Vec::<(String, u64)>::new();
while metadata.len() < file_amount {
reader.read_until(b'\n', &mut buf).await.unwrap();
let msg = String::from_utf8(buf.clone()).unwrap();
buf.clear();
// Parse 'filesize:filename'
let split = msg.split(":").collect::<Vec<&str>>();
let filesize = split[0].trim().parse::<u64>().unwrap();
let filename = split[1].trim().to_string();
metadata.push((filename, filesize));
}
println!("[INFO] Metadata: {:?}", metadata);
// Send request for each file by filename
println!("[+] Requesting files individually"); // TODO: Choose files based on input
for file in &metadata {
println!("[INFO] Current request: [{:?}]", file);
writer.write_all(file.0.as_bytes()).await.unwrap();
writer.flush().await.unwrap();
// Create file locally
let output_path = fileroot.clone() + file.0.as_str();
2022-07-10 21:16:50 +02:00
let output_file = File::create(output_path.clone()).await.unwrap();
println!("[+] New file: {}", output_path);
let mut file_buf = BufWriter::new(output_file);
// Receive the file itself
let mut remaining_data = file.1;
let mut buf = vec![0u8; buffersize];
2022-07-10 21:16:50 +02:00
while remaining_data != 0 {
if remaining_data >= buffersize as u64 {
2022-07-10 21:16:50 +02:00
let read_result = reader.read(&mut buf);
match read_result.await {
Ok(0) => {
println!("[-] Waiting for data to become available...");
sleep(Duration::from_secs(5)).await;
continue;
}
Ok(n) => {
file_buf.write_all(&mut buf).await.unwrap();
file_buf.flush().await.unwrap();
remaining_data = remaining_data - n as u64;
}
_ => {}
}
} else {
let read_result = reader.read(&mut buf);
match read_result.await {
Ok(_) => {
let mut buf_slice = &buf[0..(remaining_data as usize)];
file_buf.write_all(&mut buf_slice).await.unwrap();
file_buf.flush().await.unwrap();
remaining_data = 0;
}
_ => {}
}
}
}
// ACK file
writer.write_all(b"ACK").await.unwrap();
writer.flush().await.unwrap();
println!(
"[+] Successfully wrote {} bytes to {}\n",
file.1, output_path
);
}
println!("[+] All files finished, requesting connection termination");
2022-07-10 21:16:50 +02:00
writer.write_all(b"FIN").await.unwrap();
writer.flush().await.unwrap();
}
Ok(())
}