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,
}
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.repeat(1));
}
2 => {
id.push_str(&SPACE.repeat(1));
id.push_str(&self.fid.to_string());
id.push_str(&SPACE.repeat(1));
}
3 => {
id.push_str(&SPACE.repeat(1));
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 metadata: MetaData,
pub test: bool,
pub t_content: String,
}
impl std::fmt::Display for Question {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", &self.content.render())
}
}
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: 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;
#[derive(Debug, Deserialize)]
pub struct VerifyResult {
pub state: String,
#[serde(skip)]
pub name: String,
#[serde(skip)]
pub data_input: String,
#[serde(default)]
lang: String,
#[serde(default)]
pretty_lang: String,
#[serde(default)]
submission_id: String,
#[serde(default)]
run_success: bool,
#[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)]
std_output: String,
#[serde(flatten, default)]
info: VerifyInfo,
#[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: ".dimmed(),
&self.status.status_runtime.dimmed(),
"\n Your input: ",
&self.data_input.replace("\n", "↩"),
"\n Output: ",
ca,
"\n Expected: ",
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{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}{}.\n",
" Success\n\n".green().bold(),
" Runtime: ".dimmed(),
&self.status.status_runtime.bold(),
", faster than ",
rp.to_string().bold(),
"% ".bold(),
"of ",
&self.pretty_lang,
" online submissions for ",
&self.name,
".\n\n",
" Memory Usage: ".dimmed(),
&self.status.status_memory.bold(),
", less than ",
mp.to_string().bold(),
"% ".bold(),
"of ",
&self.pretty_lang,
" online submissions for ",
&self.name,
)
} else {
write!(
f,
"\n{}{}{}\n{}{}{}{}{}{}\n",
" Wrong Answer".red().bold(),
" Runtime: ".dimmed(),
&self.status.status_runtime.dimmed(),
"\n Your input: ",
&self.data_input.replace("\n", "↩"),
"\n Output: ",
ca,
"\n Expected: ",
eca,
)
}
}
11 => write!(
f,
"\n{}:\n\n{}{}\n{}{}\n{}{}\n",
&self.status.status_msg.red().bold(),
"Total Correct: ".green(),
&self
.analyse
.total_correct
.as_ref()
.unwrap_or(&Number::from(0))
.to_string()
.green(),
"Total Testcases: ".yellow(),
&self
.analyse
.total_testcases
.as_ref()
.unwrap_or(&Number::from(0))
.to_string()
.bold()
.yellow(),
"Last TestCase: ".dimmed(),
&self.submit.last_testcase.dimmed()
),
13 => write!(
f,
"\n{}:\n\n{:?}\n",
&self.status.status_msg.yellow().bold(),
&self.code_output,
),
14 => write!(
f,
"\n{}:\n\n{:?}\n",
&self.status.status_msg.yellow().bold(),
&self.code_output,
),
15 => write!(f, "\n{}\n", &self.status.status_msg.red().bold()),
20 => write!(
f,
"\n{}:\n\n{}\n",
&self.status.status_msg.red().bold(),
&self.error.full_compile_error
),
_ => write!(f, "{}", "\nUnknown Error...\n".red().bold()),
}
}
}
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 VerifyInfo {
#[serde(default)]
memory: i64,
#[serde(default)]
elapsed_time: i64,
#[serde(default)]
task_finish_time: i64,
}
#[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,
}
#[derive(Debug, Default, Deserialize)]
pub struct CompileError {
#[serde(default)]
compile_error: String,
#[serde(default)]
pub full_compile_error: String,
}
#[derive(Debug, Default, Deserialize)]
pub struct Expected {
#[serde(default)]
expected_status_code: i32,
#[serde(default)]
expected_lang: String,
#[serde(default)]
expected_run_success: bool,
#[serde(default)]
expected_status_runtime: String,
#[serde(default)]
expected_memory: i64,
#[serde(default, deserialize_with = "ssr")]
expected_code_output: Vec<String>,
#[serde(default)]
expected_elapsed_time: i64,
#[serde(default)]
expected_task_finish_time: i64,
#[serde(default, deserialize_with = "ssr")]
pub expected_code_answer: Vec<String>,
}
}