use super::Command;
use crate::err::Error;
use clap::{App, Arg, ArgMatches, SubCommand};
pub struct PickCommand;
static QUERY_HELP: &str = r#"Fliter questions by conditions:
Uppercase means negative
e = easy E = m+h
m = medium M = e+h
h = hard H = e+m
d = done D = not done
l = locked L = not locked
s = starred S = not starred"#;
impl Command for PickCommand {
fn usage<'a, 'pick>() -> App<'a, 'pick> {
SubCommand::with_name("pick")
.about("Pick a problem")
.visible_alias("p")
.arg(Arg::with_name("id").help("Problem id").takes_value(true))
.arg(
Arg::with_name("query")
.short("q")
.long("query")
.takes_value(true)
.help(QUERY_HELP),
)
.arg(
Arg::with_name("tag")
.short("t")
.long("tag")
.takes_value(true)
.help("Filter questions by tag"),
)
}
fn handler(m: &ArgMatches) -> Result<(), Error> {
use crate::cache::Cache;
use rand::Rng;
let cache = Cache::new()?;
let mut problems = cache.get_problems()?;
if problems.is_empty() {
cache.clone().download_problems()?;
Self::handler(m)?
}
if m.is_present("plan") {
let ids = crate::pym::exec(m.value_of("plan").unwrap_or(""))?;
crate::helper::squash(&mut problems, ids)?;
}
if m.is_present("tag") {
let ids = cache
.clone()
.get_tagged_questions(m.value_of("tag").unwrap_or(""))?;
crate::helper::squash(&mut problems, ids)?;
}
if m.is_present("query") {
let query = m.value_of("query")?;
crate::helper::filter(&mut problems, query.to_string());
}
if let Some(id) = m.value_of("id") {
problems.retain(|x| x.fid.to_string() == id);
}
let problem = &problems[rand::thread_rng().gen_range(0, problems.len())];
let r = cache.get_question(problem.fid);
if r.is_err() {
let e = r.err()?;
eprintln!("{:?}", &e);
if let Error::FeatureError(_) | Error::NetworkError(_) = e {
Self::handler(m)?;
}
} else {
println!("{}", r?);
}
Ok(())
}
}