pub use self::digit::Digit;
pub use self::file::{code_path, load_script, test_cases_path};
pub use self::filter::{filter, squash};
pub use self::html::HTML;
mod digit {
pub trait Digit<T> {
fn digit(self, d: T) -> String;
}
impl Digit<i32> for i32 {
fn digit(self, d: i32) -> String {
let mut s = self.to_string();
let space = " ".repeat((d as usize) - s.len());
s.push_str(&space);
s
}
}
impl Digit<i32> for String {
fn digit(self, d: i32) -> String {
let mut s = self.clone();
let space = " ".repeat((d as usize) - self.len());
s.push_str(&space);
s
}
}
impl Digit<i32> for &'static str {
fn digit(self, d: i32) -> String {
let mut s = self.to_string();
let space = " ".repeat((d as usize) - self.len());
s.push_str(&space);
s
}
}
}
mod filter {
use crate::cache::models::Problem;
pub fn filter(ps: &mut Vec<Problem>, query: String) {
for p in query.chars() {
match p {
'l' => ps.retain(|x| x.locked),
'L' => ps.retain(|x| !x.locked),
's' => ps.retain(|x| x.starred),
'S' => ps.retain(|x| !x.starred),
'e' => ps.retain(|x| x.level == 1),
'E' => ps.retain(|x| x.level != 1),
'm' => ps.retain(|x| x.level == 2),
'M' => ps.retain(|x| x.level != 2),
'h' => ps.retain(|x| x.level == 3),
'H' => ps.retain(|x| x.level != 3),
'd' => ps.retain(|x| x.status == "ac"),
'D' => ps.retain(|x| x.status != "ac"),
_ => {}
}
}
}
pub fn squash(ps: &mut Vec<Problem>, ids: Vec<String>) -> Result<(), crate::Error> {
use std::collections::HashMap;
let mut map: HashMap<String, bool> = HashMap::new();
ids.iter().for_each(|x| {
map.insert(x.to_string(), true).unwrap_or_default();
});
ps.retain(|x| map.get(&x.id.to_string()).is_some());
Ok(())
}
}
mod html {
use scraper::Html;
pub trait HTML {
fn render(&self) -> String;
}
impl HTML for String {
fn render(&self) -> String {
let rep = self
.replace(r#"</sup>"#, "")
.replace(r#"<sup>"#, "^")
.replace(r#"</sub>"#, "")
.replace(r#"<sub>"#, "_");
let frag = Html::parse_fragment(rep.as_str());
let res = frag
.root_element()
.text()
.fold(String::new(), |acc, e| acc + e);
res
}
}
}
mod file {
pub fn suffix(l: &str) -> Result<&'static str, crate::Error> {
match l {
"bash" => Ok("sh"),
"c" => Ok("c"),
"cpp" => Ok("cpp"),
"csharp" => Ok("cs"),
"golang" => Ok("go"),
"java" => Ok("java"),
"javascript" => Ok("js"),
"kotlin" => Ok("kt"),
"mysql" => Ok("sql"),
"php" => Ok("php"),
"python" => Ok("py"),
"python3" => Ok("py"),
"ruby" => Ok("rb"),
"rust" => Ok("rs"),
"scala" => Ok("scala"),
"swift" => Ok("swift"),
_ => Ok("c"),
}
}
use crate::{cache::models::Problem, Error};
pub fn test_cases_path(problem: &Problem) -> Result<String, Error> {
let conf = crate::cfg::locate()?;
let mut path = format!("{}/{}.tests.dat", conf.storage.code()?, conf.code.pick);
path = path.replace("${fid}", &problem.fid.to_string());
path = path.replace("${slug}", &problem.slug.to_string());
Ok(path)
}
pub fn code_path(problem: &Problem, l: Option<String>) -> Result<String, Error> {
let conf = crate::cfg::locate()?;
let mut lang = conf.code.lang;
if l.is_some() {
lang = l.ok_or(Error::NoneError)?;
}
let mut path = format!(
"{}/{}.{}",
conf.storage.code()?,
conf.code.pick,
suffix(&lang)?,
);
path = path.replace("${fid}", &problem.fid.to_string());
path = path.replace("${slug}", &problem.slug.to_string());
Ok(path)
}
pub fn load_script(module: &str) -> Result<String, crate::Error> {
use std::fs::File;
use std::io::Read;
let conf = crate::cfg::locate()?;
let mut script = "".to_string();
File::open(format!("{}/{}.py", conf.storage.scripts()?, module))?
.read_to_string(&mut script)?;
Ok(script)
}
}