futriix-cli/src/main.rs

157 lines
4.3 KiB
Rust
Raw Normal View History

2025-05-25 14:35:35 +00:00
use std::io::{self, Write};
use std::net::{TcpStream, ToSocketAddrs};
use std::time::Duration;
use std::env;
use colored::*;
use regex::Regex;
mod resp;
const DEFAULT_PORT: u16 = 9880;
const PROMPT_NAME: &str = "futriix";
const CONNECTION_TIMEOUT: u64 = 5; // seconds
fn main() {
// Parse command line arguments
let args: Vec<String> = env::args().collect();
let mut host = "127.0.0.1".to_string();
let mut port = DEFAULT_PORT;
if args.len() > 1 {
let re = Regex::new(r"^(?:([^:]+):)?([^:]+)(?::(\d+))?$").unwrap();
if let Some(caps) = re.captures(&args[1]) {
if let Some(h) = caps.get(2) {
host = h.as_str().to_string();
}
if let Some(p) = caps.get(3) {
port = p.as_str().parse().unwrap_or(DEFAULT_PORT);
}
}
}
// Try to connect to server
let addr = format!("{}:{}", host, port);
2025-05-25 20:31:46 +00:00
let socket_addr = match format!("{}:{}", host, port).to_socket_addrs() {
Ok(mut addrs) => addrs.next().expect("No address found"),
Err(_e) => {
eprintln!("Failed to resolve address {}", addr);
return;
}
};
2025-05-25 14:35:35 +00:00
let stream = match TcpStream::connect_timeout(&socket_addr, Duration::from_secs(CONNECTION_TIMEOUT)) {
Ok(s) => s,
2025-05-25 20:31:46 +00:00
Err(_e) => {
eprintln!("Connection to port {} refused", port);
2025-05-25 14:35:35 +00:00
return;
}
};
println!("Connected to {}", addr);
// Main REPL loop
let mut input = String::new();
loop {
// Print prompt
print_prompt(&host, port);
2025-05-25 20:40:33 +00:00
// Read input
2025-05-25 14:35:35 +00:00
input.clear();
io::stdin().read_line(&mut input).expect("Failed to read input");
let input = input.trim();
if input.is_empty() {
continue;
}
// Handle special commands
if input.eq_ignore_ascii_case("quit") || input.eq_ignore_ascii_case("exit") {
break;
}
2025-05-25 20:31:46 +00:00
// Validate command before sending
if !is_valid_command(input) {
eprintln!("Error: Invalid command format");
continue;
}
2025-05-25 14:35:35 +00:00
// Send command to server
match send_command(&stream, input) {
Ok(response) => {
print_response(&response);
}
Err(e) => {
2025-05-25 20:40:33 +00:00
eprintln!("Error: {}", e.to_string().replace("KeyDB", "Futriix"));
2025-05-25 20:31:46 +00:00
// Check if connection was lost
if e.kind() == io::ErrorKind::ConnectionAborted ||
e.kind() == io::ErrorKind::ConnectionReset {
eprintln!("Connection lost. Please restart the client.");
break;
}
2025-05-25 14:35:35 +00:00
}
}
2025-05-24 00:26:45 +03:00
}
2025-05-25 14:35:35 +00:00
}
2025-05-25 20:31:46 +00:00
fn is_valid_command(cmd: &str) -> bool {
if cmd.is_empty() {
return false;
}
if cmd.chars().any(|c| c.is_control()) {
return false;
}
cmd.split_whitespace().next().is_some()
}
2025-05-25 14:35:35 +00:00
fn print_prompt(host: &str, port: u16) {
let prompt = format!("{}:{}:{}:~>", PROMPT_NAME, host, port);
print!("{} ", prompt.green());
io::stdout().flush().unwrap();
}
2025-05-25 20:31:46 +00:00
fn send_command(stream: &TcpStream, command: &str) -> io::Result<resp::Value> {
2025-05-25 14:35:35 +00:00
let parts: Vec<&str> = command.split_whitespace().collect();
let mut resp_command = String::new();
2025-05-24 00:26:45 +03:00
2025-05-25 14:35:35 +00:00
resp_command.push_str(&format!("*{}\r\n", parts.len()));
for part in parts {
resp_command.push_str(&format!("${}\r\n{}\r\n", part.len(), part));
}
let mut stream = stream.try_clone()?;
stream.write_all(resp_command.as_bytes())?;
2025-05-25 20:31:46 +00:00
let mut decoder = resp::Decoder::new(&stream);
2025-05-25 14:35:35 +00:00
decoder.decode()
2025-05-24 00:26:45 +03:00
}
2025-05-25 14:35:35 +00:00
2025-05-25 20:31:46 +00:00
fn print_response(value: &resp::Value) {
2025-05-25 14:35:35 +00:00
match value {
2025-05-25 20:40:33 +00:00
resp::Value::SimpleString(s) | resp::Value::BulkString(s) => {
println!("{}", s.replace("KeyDB", "Futriix"))
},
resp::Value::Error(e) => {
println!("(error) {}", e.replace("KeyDB", "Futriix"))
},
2025-05-25 20:31:46 +00:00
resp::Value::Integer(i) => println!("(integer) {}", i),
resp::Value::Array(arr) => {
2025-05-25 14:35:35 +00:00
for (i, item) in arr.iter().enumerate() {
print!("{}) ", i + 1);
print_response(item);
}
}
2025-05-25 20:31:46 +00:00
resp::Value::Null => println!("(nil)"),
2025-05-25 14:35:35 +00:00
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_print_prompt() {
print_prompt("127.0.0.1", 9880);
}
}