Created
May 2, 2022 22:46
-
-
Save arjunsk/cfb1674de5987a0fc621d578b969950e to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
use mini_redis::{client, DEFAULT_PORT}; | |
use bytes::Bytes; | |
use std::num::ParseIntError; | |
use std::str; | |
use std::time::Duration; | |
use structopt::StructOpt; | |
#[derive(StructOpt, Debug)] | |
#[structopt(name = "mini-redis-cli", version = env!("CARGO_PKG_VERSION"), author = env!("CARGO_PKG_AUTHORS"), about = "Issue Redis commands")] | |
struct Cli { | |
#[structopt(subcommand)] | |
command: Command, | |
#[structopt(name = "hostname", long = "--host", default_value = "127.0.0.1")] | |
host: String, | |
#[structopt(name = "port", long = "--port", default_value = DEFAULT_PORT)] | |
port: String, | |
} | |
#[derive(StructOpt, Debug)] | |
enum Command { | |
/// Get the value of key. | |
Get { | |
/// Name of key to get | |
key: String, | |
}, | |
/// Set key to hold the string value. | |
Set { | |
/// Name of key to set | |
key: String, | |
/// Value to set. | |
#[structopt(parse(from_str = bytes_from_str))] | |
value: Bytes, | |
/// Expire the value after specified amount of time | |
#[structopt(parse(try_from_str = duration_from_ms_str))] | |
expires: Option<Duration>, | |
}, | |
/// Publisher to send a message to a specific channel. | |
Publish { | |
/// Name of channel | |
channel: String, | |
#[structopt(parse(from_str = bytes_from_str))] | |
/// Message to publish | |
message: Bytes, | |
}, | |
/// Subscribe a client to a specific channel or channels. | |
Subscribe { | |
/// Specific channel or channels | |
channels: Vec<String>, | |
}, | |
} | |
/// Entry point for CLI tool. | |
/// | |
/// The `[tokio::main]` annotation signals that the Tokio runtime should be | |
/// started when the function is called. The body of the function is executed | |
/// within the newly spawned runtime. | |
/// | |
/// `flavor = "current_thread"` is used here to avoid spawning background | |
/// threads. The CLI tool use case benefits more by being lighter instead of | |
/// multi-threaded. | |
#[tokio::main(flavor = "current_thread")] | |
async fn main() -> mini_redis::Result<()> { | |
// Enable logging | |
tracing_subscriber::fmt::try_init()?; | |
// Parse command line arguments | |
let cli = Cli::from_args(); | |
// Get the remote address to connect to | |
let addr = format!("{}:{}", cli.host, cli.port); | |
// Establish a connection | |
let mut client = client::connect(&addr).await?; | |
// Process the requested command | |
match cli.command { | |
Command::Get { key } => { | |
if let Some(value) = client.get(&key).await? { | |
if let Ok(string) = str::from_utf8(&value) { | |
println!("\"{}\"", string); | |
} else { | |
println!("{:?}", value); | |
} | |
} else { | |
println!("(nil)"); | |
} | |
} | |
Command::Set { | |
key, | |
value, | |
expires: None, | |
} => { | |
client.set(&key, value).await?; | |
println!("OK"); | |
} | |
Command::Set { | |
key, | |
value, | |
expires: Some(expires), | |
} => { | |
client.set_expires(&key, value, expires).await?; | |
println!("OK"); | |
} | |
Command::Publish { channel, message } => { | |
client.publish(&channel, message.into()).await?; | |
println!("Publish OK"); | |
} | |
Command::Subscribe { channels } => { | |
if channels.is_empty() { | |
return Err("channel(s) must be provided".into()); | |
} | |
let mut subscriber = client.subscribe(channels).await?; | |
// await messages on channels | |
while let Some(msg) = subscriber.next_message().await? { | |
println!( | |
"got message from the channel: {}; message = {:?}", | |
msg.channel, msg.content | |
); | |
} | |
} | |
} | |
Ok(()) | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment