more code commants and reorganization for better redability
This commit is contained in:
parent
0d06438230
commit
4110778123
36
src/main.rs
36
src/main.rs
@ -1,13 +1,14 @@
|
|||||||
use tokio::net::{TcpListener, TcpStream};
|
use std::collections::HashMap;
|
||||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
|
||||||
use std::env;
|
use std::env;
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
use std::collections::HashMap;
|
|
||||||
use std::net::SocketAddr;
|
use std::net::SocketAddr;
|
||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
|
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
||||||
|
use tokio::net::{TcpListener, TcpStream};
|
||||||
|
|
||||||
#[tokio::main]
|
#[tokio::main]
|
||||||
async fn main() -> Result<(), Box<dyn Error>> {
|
async fn main() -> Result<(), Box<dyn Error>> {
|
||||||
|
// Parse command-line arguments
|
||||||
let args: Vec<String> = env::args().collect();
|
let args: Vec<String> = env::args().collect();
|
||||||
if args.len() != 3 {
|
if args.len() != 3 {
|
||||||
println!("Usage: {} [server|client] [addr:port]", args[0]);
|
println!("Usage: {} [server|client] [addr:port]", args[0]);
|
||||||
@ -27,52 +28,59 @@ async fn main() -> Result<(), Box<dyn Error>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn start_server(addr: &str) -> Result<(), Box<dyn Error>> {
|
async fn start_server(addr: &str) -> Result<(), Box<dyn Error>> {
|
||||||
|
// Start the TCP listener
|
||||||
let listener = TcpListener::bind(addr).await?;
|
let listener = TcpListener::bind(addr).await?;
|
||||||
|
|
||||||
println!("Server listening on {}", addr);
|
println!("Server listening on {}", addr);
|
||||||
|
|
||||||
|
// Shared data for client connections
|
||||||
let clients: Arc<Mutex<HashMap<SocketAddr, String>>> = Arc::new(Mutex::new(HashMap::new()));
|
let clients: Arc<Mutex<HashMap<SocketAddr, String>>> = Arc::new(Mutex::new(HashMap::new()));
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
|
// Accept a new client connection
|
||||||
let (mut socket, client_addr) = listener.accept().await?;
|
let (mut socket, client_addr) = listener.accept().await?;
|
||||||
|
|
||||||
|
// Read username from the client
|
||||||
let mut buf = vec![0; 1024];
|
let mut buf = vec![0; 1024];
|
||||||
socket.read(&mut buf).await?;
|
socket.read(&mut buf).await?;
|
||||||
|
|
||||||
let username = String::from_utf8(buf.clone()).unwrap();
|
let username = String::from_utf8(buf.clone()).unwrap();
|
||||||
|
|
||||||
println!("Accepted connection from {}", client_addr);
|
println!("Accepted connection from {}", client_addr);
|
||||||
|
|
||||||
let clients_shared = Arc::clone(&clients);
|
let clients_shared = Arc::clone(&clients);
|
||||||
|
|
||||||
|
// Insert client into the shared HashMap
|
||||||
clients_shared.lock().unwrap().insert(client_addr, username);
|
clients_shared.lock().unwrap().insert(client_addr, username);
|
||||||
|
|
||||||
|
// Spawn a new task for each client connection
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
let mut buf = vec![0; 1024];
|
let mut buf = vec![0; 1024];
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let n = match socket.read(&mut buf).await {
|
let n = match socket.read(&mut buf).await {
|
||||||
Ok(n) if n == 0 => {
|
Ok(n) if n == 0 => {
|
||||||
|
// Client disconnected
|
||||||
let username = clients_shared.lock().unwrap().remove(&client_addr).unwrap_or("Unknown".to_string());
|
let username = clients_shared.lock().unwrap().remove(&client_addr).unwrap_or("Unknown".to_string());
|
||||||
println!("Client {} ({}) disconnected!", username, client_addr);
|
println!("Client {} ({}) disconnected!", username, client_addr);
|
||||||
return;
|
return;
|
||||||
},
|
}
|
||||||
Ok(n) => n,
|
Ok(n) => n,
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
eprintln!("failed to read from socket; err = {:?}", e);
|
eprintln!("Failed to read from socket; err = {:?}", e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let username = clients_shared.lock().unwrap().get(&client_addr).unwrap_or(&"Unknown".to_string()).clone();
|
let username = clients_shared.lock().unwrap().get(&client_addr).unwrap_or(&"Unknown".to_string()).clone();
|
||||||
|
|
||||||
println!("Received message from {} ({}): {}",
|
println!(
|
||||||
|
"Received message from {} ({}): {}",
|
||||||
username,
|
username,
|
||||||
client_addr,
|
client_addr,
|
||||||
String::from_utf8_lossy(&buf[..n]));
|
String::from_utf8_lossy(&buf[..n])
|
||||||
|
);
|
||||||
|
|
||||||
if let Err(e) = socket.write_all(&buf[0..n]).await {
|
if let Err(e) = socket.write_all(&buf[0..n]).await {
|
||||||
eprintln!("failed to write to socket; err = {:?}", e);
|
eprintln!("Failed to write to socket; err = {:?}", e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -81,12 +89,14 @@ async fn start_server(addr: &str) -> Result<(), Box<dyn Error>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fn start_client(addr: &str) -> Result<(), Box<dyn Error>> {
|
async fn start_client(addr: &str) -> Result<(), Box<dyn Error>> {
|
||||||
|
// Connect to the server
|
||||||
let mut stream = TcpStream::connect(addr).await?;
|
let mut stream = TcpStream::connect(addr).await?;
|
||||||
|
|
||||||
println!("Enter a username: ");
|
println!("Enter a username: ");
|
||||||
let mut username = String::new();
|
let mut username = String::new();
|
||||||
std::io::stdin().read_line(&mut username)?;
|
std::io::stdin().read_line(&mut username)?;
|
||||||
|
|
||||||
|
// Send username to the server
|
||||||
stream.write_all(username.as_bytes()).await?;
|
stream.write_all(username.as_bytes()).await?;
|
||||||
|
|
||||||
println!("Connected to server at {}", addr);
|
println!("Connected to server at {}", addr);
|
||||||
|
Loading…
Reference in New Issue
Block a user