use super::schemas::{problems, tags};
use crate::helper::HTML;
use colored::Colorize;
use serde::{Deserialize, Serialize};
use serde_json::Number;
#[derive(Clone, Insertable, Queryable, Serialize, Debug)]
#[table_name = "tags"]
pub struct Tag {
pub tag: String,
pub refs: String,
}
#[derive(AsChangeset, Clone, Identifiable, Insertable, Queryable, Serialize, Debug)]
#[table_name = "problems"]
pub struct Problem {
pub category: String,
pub fid: i32,
pub id: i32,
pub level: i32,
pub locked: bool,
pub name: String,
pub percent: f32,
pub slug: String,
pub starred: bool,
pub status: String,
pub desc: String,
}
impl Problem {
fn display_level(&self) -> &str {
match self.level {
1 => "Easy",
2 => "Medium",
3 => "Hard",
_ => "Unknown",
}
}
pub fn desc_comment(&self) -> String {
let mut res = String::new();
res += format!("// Category: {}\n", self.category).as_str();
res += format!("// Level: {}\n", self.display_level(),).as_str();
res += format!("// Percent: {}%\n\n", self.percent).as_str();
res + "\n"
}
}
static DONE: &str = " ✔";
static ETC: &str = "...";
static LOCK: &str = "🔒";
static NDONE: &str = "✘";
static SPACE: &str = " ";
impl std::fmt::Display for Problem {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let space_2 = SPACE.repeat(2);
let mut lock = space_2.as_str();
let mut done = space_2.normal();
let mut id = "".to_string();
let mut name = "".to_string();
let mut level = "".normal();
if self.locked {
lock = LOCK
};
if self.status == "ac" {
done = DONE.green().bold();
} else if self.status == "notac" {
done = NDONE.green().bold();
}
match self.fid.to_string().len() {
1 => {
id.push_str(&SPACE.repeat(2));
id.push_str(&self.fid.to_string());
id.push_str(SPACE);
}
2 => {
id.push_str(SPACE);
id.push_str(&self.fid.to_string());
id.push_str(SPACE);
}
3 => {
id.push_str(SPACE);
id.push_str(&self.fid.to_string());
}
4 => {
id.push_str(&self.fid.to_string());
}
_ => {
id.push_str(&space_2);
id.push_str(&space_2);
}
}
if self.name.len() < 60_usize {
name.push_str(&self.name);
name.push_str(&SPACE.repeat(60 - &self.name.len()));
} else {
name.push_str(&self.name[..49]);
name = name.trim_end().to_string();
name.push_str(ETC);
name.push_str(&SPACE.repeat(60 - name.len()));
}
level = match self.level {
1 => "Easy ".bright_green(),
2 => "Medium".bright_yellow(),
3 => "Hard ".bright_red(),
_ => level,
};
let mut pct = self.percent.to_string();
if pct.len() < 5 {
pct.push_str(&"0".repeat(5 - pct.len()));
}
write!(
f,
" {} {} [{}] {} {} ({} %)",
lock,
done,
id,
name,
level,
&pct[..5]
)
}
}
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct Question {
pub content: String,
pub stats: Stats,
pub defs: CodeDefintion,
pub case: String,
pub all_cases: String,
pub metadata: MetaData,
pub test: bool,
pub t_content: String,
}
impl Question {
pub fn desc(&self) -> String {
self.content.render()
}
pub fn desc_comment(&self) -> String {
let desc = self.content.render();
let mut res = desc
.lines()
.fold("/*\n".to_string(), |acc, e| acc + " * " + e + "\n");
res += " */\n";
res
}
}
use question::*;
mod question {
use serde::{Deserialize, Serialize};
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct CodeDefintion(pub Vec<CodeDefintionInner>);
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct CodeDefintionInner {
pub value: String,
pub text: String,
#[serde(alias = "defaultCode")]
pub code: String,
}
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct Stats {
#[serde(alias = "totalAccepted")]
tac: String,
#[serde(alias = "totalSubmission")]
tsm: String,
#[serde(alias = "totalAcceptedRaw")]
tacr: i32,
#[serde(alias = "totalSubmissionRaw")]
tsmr: i32,
#[serde(alias = "acRate")]
rate: String,
}
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct MetaData {
pub name: Option<String>,
pub params: Option<Vec<Param>>,
pub r#return: Return,
}
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct Param {
pub name: String,
pub r#type: String,
}
#[derive(Debug, Default, Serialize, Deserialize)]
pub struct Return {
pub r#type: String,
}
}
#[derive(Debug, Deserialize)]
pub struct RunCode {
#[serde(default)]
pub interpret_id: String,
#[serde(default)]
pub test_case: String,
#[serde(default)]
pub submission_id: i64,
}
use super::parser::ssr;
use crate::cache::Run;
#[derive(Default, Debug, Deserialize)]
pub struct VerifyResult {
pub state: String,
#[serde(skip)]
pub name: String,
#[serde(skip)]
pub data_input: String,
#[serde(skip)]
pub result_type: Run,
#[serde(default)]
pretty_lang: String,
#[serde(default)]
correct_answer: bool,
#[serde(default, deserialize_with = "ssr")]
code_answer: Vec<String>,
#[serde(default, deserialize_with = "ssr")]
code_output: Vec<String>,
#[serde(default, deserialize_with = "ssr")]
expected_output: Vec<String>,
#[serde(default)]
std_output: String,
#[serde(flatten, default)]
status: VerifyStatus,
#[serde(flatten, default)]
analyse: Analyse,
#[serde(flatten, default)]
expected: Expected,
#[serde(flatten, default)]
error: CompileError,
#[serde(flatten, default)]
submit: Submit,
}
impl std::fmt::Display for VerifyResult {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let ca = match &self.code_answer.len() {
1 => self.code_answer[0].to_string(),
_ => self.code_answer.join("↩ "),
};
let eca = match &self.expected.expected_code_answer.len() {
1 => self.expected.expected_code_answer[0].to_string(),
_ => self.expected.expected_code_answer.join("↩ "),
};
debug!("{:#?}", &self);
match &self.status.status_code {
10 => {
if self.correct_answer {
write!(
f,
"\n{}{}{}\n{}{}{}{}{}{}\n",
&self.status.status_msg.green().bold(),
&"Runtime: ".before_spaces(7).dimmed(),
&self.status.status_runtime.dimmed(),
&"\nYour input:".after_spaces(4),
&self.data_input.replace('\n', "↩ "),
&"\nOutput:".after_spaces(8),
ca,
&"\nExpected:".after_spaces(6),
eca,
)?
} else if !self.submit.compare_result.is_empty() {
let cache = super::Cache::new().expect("cache gen failed");
cache
.update_after_ac(
self.submit
.question_id
.parse()
.expect("submit succcessfully, parse question_id to i32 failed"),
)
.expect("update ac to cache failed");
let (mut rp, mut mp) = (0, 0);
if let Some(n) = &self.analyse.runtime_percentile {
if n.is_f64() {
rp = n.as_f64().unwrap_or(0.0) as i64;
} else {
rp = n.as_i64().unwrap_or(0);
}
}
if let Some(n) = &self.analyse.memory_percentile {
if n.is_f64() {
mp = n.as_f64().unwrap_or(0.0) as i64;
} else {
mp = n.as_i64().unwrap_or(0);
}
}
write!(
f,
"\n{}{}{}\
, faster than \
{}{}\
of \
{} \
online submissions for \
{}.\n\n\
{}{}\
, less than \
{}{}\
of \
{} {}.\n\n",
"Success\n\n".green().bold(),
"Runtime: ".dimmed(),
&self.status.status_runtime.bold(),
rp.to_string().bold(),
"% ".bold(),
&self.pretty_lang,
&self.name,
"Memory Usage: ".dimmed(),
&self.status.status_memory.bold(),
mp.to_string().bold(),
"% ".bold(),
&self.pretty_lang,
&self.name,
)?
} else {
write!(
f,
"\n{}{}{}\n{}{}{}{}{}{}\n",
"Wrong Answer".red().bold(),
" Runtime: ".dimmed(),
&self.status.status_runtime.dimmed(),
&"\nYour input:".after_spaces(4),
&self.data_input.replace('\n', "↩ "),
&"\nOutput:".after_spaces(8),
ca,
&"\nExpected:".after_spaces(6),
eca,
)?
}
}
11 => write!(
f,
"\n{}\n\n{}{}\n{}{}\n{}{}{}{}{}{}\n",
&self.status.status_msg.red().bold(),
"Cases passed:".after_spaces(2).green(),
&self
.analyse
.total_correct
.as_ref()
.unwrap_or(&Number::from(0))
.to_string()
.green(),
&"Total cases:".after_spaces(3).yellow(),
&self
.analyse
.total_testcases
.as_ref()
.unwrap_or(&Number::from(0))
.to_string()
.bold()
.yellow(),
&"Last case:".after_spaces(5).dimmed(),
&self.submit.last_testcase.replace('\n', "↩ ").dimmed(),
&"\nOutput:".after_spaces(8),
self.code_output[0],
&"\nExpected:".after_spaces(6),
self.expected_output[0],
)?,
12 => write!(
f,
"\n{}\n\n{}{}\n",
&self.status.status_msg.yellow().bold(),
&"Last case:".after_spaces(5).dimmed(),
&self.data_input.replace('\n', "↩ "),
)?,
13 | 14 => write!(f, "\n{}\n", &self.status.status_msg.yellow().bold(),)?,
15 => write!(
f,
"\n{}\n{}\n'",
&self.status.status_msg.red().bold(),
&self.status.runtime_error
)?,
20 => write!(
f,
"\n{}:\n\n{}\n",
&self.status.status_msg.red().bold(),
&self.error.full_compile_error.dimmed()
)?,
_ => write!(
f,
"{}{}{}{}{}{}{}{}",
"\nUnknown Error...\n".red().bold(),
"\nBingo! Welcome to fix this! Pull your request at ".yellow(),
"https://github.com/clearloop/leetcode-cli/pulls"
.dimmed()
.underline(),
", and this file is located at ".yellow(),
"leetcode-cli/src/cache/models.rs".dimmed().underline(),
" waiting for you! Yep, line ".yellow(),
"385".dimmed().underline(),
".\n".yellow(),
)?,
};
match &self.result_type {
Run::Test => {
if !self.code_output.is_empty() {
write!(
f,
"{}{}",
&"Stdout:".after_spaces(8).purple(),
&self.code_output.join(&"\n".after_spaces(15))
)
} else {
write!(f, "")
}
}
_ => {
if !self.std_output.is_empty() {
write!(
f,
"{}{}",
&"Stdout:".after_spaces(8).purple(),
&self.std_output.replace('\n', &"\n".after_spaces(15))
)
} else {
write!(f, "")
}
}
}
}
}
use verify::*;
mod verify {
use super::super::parser::ssr;
use serde::Deserialize;
use serde_json::Number;
#[derive(Debug, Default, Deserialize)]
pub struct Submit {
#[serde(default)]
pub question_id: String,
#[serde(default)]
pub last_testcase: String,
#[serde(default)]
pub compare_result: String,
}
#[derive(Debug, Default, Deserialize)]
pub struct Analyse {
#[serde(default)]
pub total_correct: Option<Number>,
#[serde(default)]
pub total_testcases: Option<Number>,
#[serde(default)]
pub runtime_percentile: Option<Number>,
#[serde(default)]
pub memory_percentile: Option<Number>,
}
#[derive(Debug, Default, Deserialize)]
pub struct VerifyStatus {
#[serde(default)]
pub status_code: i32,
#[serde(default)]
pub status_msg: String,
#[serde(default)]
pub status_memory: String,
#[serde(default)]
pub status_runtime: String,
#[serde(default)]
pub runtime_error: String,
}
#[derive(Debug, Default, Deserialize)]
pub struct CompileError {
#[serde(default)]
pub full_compile_error: String,
}
#[derive(Debug, Default, Deserialize)]
pub struct Expected {
#[serde(default, deserialize_with = "ssr")]
pub expected_code_answer: Vec<String>,
}
}
trait Formatter {
fn after_spaces(&self, spaces: usize) -> String;
fn before_spaces(&self, spaces: usize) -> String;
}
impl Formatter for str {
fn after_spaces(&self, spaces: usize) -> String {
let mut r = String::new();
r.push_str(self);
r.push_str(&" ".repeat(spaces));
r
}
fn before_spaces(&self, spaces: usize) -> String {
let mut r = String::new();
r.push_str(&" ".repeat(spaces));
r.push_str(self);
r
}
}