protohackers/problem_01/src/main.rs

129 lines
4.1 KiB
Rust
Raw Normal View History

2023-04-20 08:58:47 +00:00
use primes::is_prime;
use serde::{Deserialize, Serialize};
use tokio::io::AsyncBufReadExt;
use tokio::io::AsyncWriteExt;
use tokio::io::BufReader;
use tokio::net::{TcpListener, TcpStream};
const IS_PRIME: &str = "isPrime";
const MAL_FORMAT: &str = "}mal";
#[derive(Debug, Deserialize, Serialize)]
struct Request {
method: String,
2023-04-20 18:28:19 +00:00
number: serde_json::value::Number,
2023-04-20 08:58:47 +00:00
}
#[derive(Debug, Deserialize, Serialize)]
struct Response {
method: String,
prime: bool,
}
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
env_logger::init();
2023-04-20 15:57:56 +00:00
let listener = TcpListener::bind("0.0.0.0:1222").await?;
2023-04-20 08:58:47 +00:00
log::info!("Start TCP server");
loop {
let (socket, _) = listener.accept().await?;
tokio::spawn(async move {
log::info!("Handle incoming request");
let _ = handle_request(socket).await;
});
}
}
async fn handle_request(mut socket: TcpStream) {
let (read, mut write) = socket.split();
let mut buf: Vec<u8> = Vec::new();
let mut reader = BufReader::new(read);
2023-04-20 18:44:22 +00:00
loop {
let bytes = reader.read_until(b'\n', &mut buf).await;
2023-04-20 18:28:19 +00:00
2023-04-20 18:44:22 +00:00
if let Ok(0) = bytes {
return;
}
2023-04-20 18:28:19 +00:00
2023-04-20 18:44:22 +00:00
if let Ok(1) = bytes {
return;
}
2023-04-20 15:57:56 +00:00
2023-04-20 18:44:22 +00:00
let message: serde_json::Value = match serde_json::from_slice(&buf) {
Ok(m) => m,
Err(e) => {
let _ = write.write(&MAL_FORMAT.as_bytes()).await;
let _ = write.write(&[b'\n']).await;
let _ = write.flush().await;
return;
}
};
log::info!("Message received: {}", message);
match serde_json::from_slice::<Request>(&buf) {
Ok(m) => {
let possible_prime = match m.number.to_string().parse::<u64>() {
Ok(n) => n,
Err(e) => {
log::error!("Not a valid number: {}", e);
if let Err(e) = write
.write(&serde_json::to_string(&Response {method: IS_PRIME.to_owned(), prime: false}).unwrap().as_bytes())
.await
{
log::error!("Error writing serialize step: {}", e);
}
return;
}
};
let res = Response {
method: IS_PRIME.to_owned(),
prime: is_prime(possible_prime),
};
if m.method == IS_PRIME.to_owned() {
log::info!("All good, send response: {:?}", res);
if let Err(e) = write
.write(&serde_json::to_string(&res).unwrap().as_bytes())
.await
{
log::error!("Error writing serialize step: {}", e);
}
if let Err(e) = write.write(&[b'\n']).await {
log::error!("Error writing: {}", e);
}
if let Err(e) = write.flush().await {
log::error!("Error flushing: {}", e);
}
} else {
log::error!("Method is not isPrime");
if let Err(e) = write.write(&MAL_FORMAT.as_bytes()).await {
log::error!("Write mal_format failed!");
}
if let Err(e) = write.write(&[b'\n']).await {
log::error!("Error writing escape character!");
}
if let Err(e) = write.flush().await {
log::error!("Error flushing socket!");
}
log::info!("Wrote malformat response");
2023-04-20 15:57:56 +00:00
}
2023-04-20 08:58:47 +00:00
}
2023-04-20 18:44:22 +00:00
Err(e) => {
log::error!("Error parsing the message: {}", e);
log::error!("Message: {}", String::from_utf8_lossy(&buf));
2023-04-20 08:58:47 +00:00
2023-04-20 18:44:22 +00:00
let _ = write.write(&MAL_FORMAT.as_bytes()).await;
let _ = write.write(&[b'\n']).await;
let _ = write.flush().await;
}
2023-04-20 08:58:47 +00:00
}
}
2023-04-17 19:45:27 +00:00
}