Some checks failed
actionlint check / actionlint check (pull_request) Successful in 7s
checkov check / checkov check (pull_request) Successful in 53s
conventional commit messages check / conventional commit messages check (pull_request) Successful in 4s
conventional pull request title check / conventional pull request title check (pull_request) Successful in 2s
dotenv-linter check / dotenv-linter check (pull_request) Successful in 6s
GitLeaks check / GitLeaks check (pull_request) Successful in 7s
hadolint check / hadolint check (pull_request) Successful in 9s
htmlhint check / htmlhint check (pull_request) Successful in 9s
markdownlint check / markdownlint check (pull_request) Successful in 9s
Prettier check / Prettier check (pull_request) Successful in 10s
Rust check / Rust check (pull_request) Failing after 7m40s
ShellCheck check / ShellCheck check (pull_request) Successful in 12s
Stylelint check / Stylelint check (pull_request) Successful in 11s
yamllint check / yamllint check (pull_request) Successful in 14s
179 lines
4.6 KiB
Rust
179 lines
4.6 KiB
Rust
use crate::models::category::Category;
|
|
use crate::models::subtask::Subtask;
|
|
#[cfg(feature = "server")]
|
|
use crate::schema::tasks;
|
|
use crate::utils::reverse_ord_option::ReverseOrdOption;
|
|
use chrono::NaiveDateTime;
|
|
#[cfg(feature = "server")]
|
|
use diesel::prelude::*;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::cmp::Ordering;
|
|
use validator::Validate;
|
|
|
|
const TITLE_LENGTH_MIN: u64 = 1;
|
|
const TITLE_LENGTH_MAX: u64 = 255;
|
|
|
|
#[derive(Serialize, Deserialize, PartialEq, Clone, Debug)]
|
|
#[cfg_attr(feature = "server", derive(Queryable, Selectable, Identifiable))]
|
|
#[cfg_attr(feature = "server", diesel(table_name = tasks, check_for_backend(diesel::pg::Pg)))]
|
|
pub struct Task {
|
|
id: i32,
|
|
title: String,
|
|
deadline: Option<chrono::NaiveDate>,
|
|
category: Category,
|
|
project_id: Option<i32>,
|
|
created_at: NaiveDateTime,
|
|
updated_at: NaiveDateTime,
|
|
}
|
|
|
|
impl Task {
|
|
pub fn id(&self) -> i32 {
|
|
self.id
|
|
}
|
|
|
|
pub fn title(&self) -> &str {
|
|
&self.title
|
|
}
|
|
|
|
pub fn deadline(&self) -> Option<chrono::NaiveDate> {
|
|
self.deadline
|
|
}
|
|
|
|
pub fn category(&self) -> &Category {
|
|
&self.category
|
|
}
|
|
|
|
pub fn project_id(&self) -> Option<i32> {
|
|
self.project_id
|
|
}
|
|
|
|
pub fn created_at(&self) -> NaiveDateTime {
|
|
self.created_at
|
|
}
|
|
|
|
pub fn updated_at(&self) -> NaiveDateTime {
|
|
self.updated_at
|
|
}
|
|
}
|
|
|
|
impl Eq for Task {}
|
|
|
|
impl PartialOrd<Self> for Task {
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
|
Some(self.cmp(other))
|
|
}
|
|
}
|
|
|
|
impl Ord for Task {
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
match (&self.category, &other.category) {
|
|
(Category::Inbox, Category::Inbox) => self.created_at.cmp(&other.created_at),
|
|
(
|
|
Category::Calendar {
|
|
date: self_date,
|
|
time: self_time,
|
|
..
|
|
},
|
|
Category::Calendar {
|
|
date: other_date,
|
|
time: other_time,
|
|
..
|
|
},
|
|
) => self_date
|
|
.cmp(other_date)
|
|
.then(
|
|
ReverseOrdOption::from(
|
|
&self_time.as_ref().map(|calendar_time| calendar_time.time()),
|
|
)
|
|
.cmp(&ReverseOrdOption::from(
|
|
&other_time
|
|
.as_ref()
|
|
.map(|calendar_time| calendar_time.time()),
|
|
)),
|
|
)
|
|
.then(
|
|
ReverseOrdOption::from(&self.deadline())
|
|
.cmp(&ReverseOrdOption::from(&other.deadline())),
|
|
)
|
|
.then(self.created_at.cmp(&other.created_at)),
|
|
(Category::Done, Category::Done) | (Category::Trash, Category::Trash) => {
|
|
self.updated_at.cmp(&other.updated_at).reverse()
|
|
}
|
|
(_, _) => ReverseOrdOption::from(&self.deadline())
|
|
.cmp(&ReverseOrdOption::from(&other.deadline()))
|
|
.then(self.created_at.cmp(&other.created_at)),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize, PartialEq, Clone, Debug)]
|
|
pub struct TaskWithSubtasks {
|
|
task: Task,
|
|
subtasks: Vec<Subtask>,
|
|
}
|
|
|
|
impl TaskWithSubtasks {
|
|
pub fn new(task: Task, subtasks: Vec<Subtask>) -> Self {
|
|
Self { task, subtasks }
|
|
}
|
|
|
|
pub fn task(&self) -> &Task {
|
|
&self.task
|
|
}
|
|
|
|
pub fn subtasks(&self) -> &Vec<Subtask> {
|
|
&self.subtasks
|
|
}
|
|
}
|
|
|
|
impl Eq for TaskWithSubtasks {}
|
|
|
|
impl PartialOrd<Self> for TaskWithSubtasks {
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
|
Some(self.cmp(other))
|
|
}
|
|
}
|
|
|
|
impl Ord for TaskWithSubtasks {
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
self.task().cmp(other.task())
|
|
}
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize, Validate, Clone, Debug)]
|
|
#[cfg_attr(feature = "server", derive(Insertable))]
|
|
#[cfg_attr(feature = "server", diesel(table_name = tasks))]
|
|
pub struct NewTask {
|
|
#[validate(length(
|
|
min = "TITLE_LENGTH_MIN",
|
|
max = "TITLE_LENGTH_MAX",
|
|
code = "title_length"
|
|
))]
|
|
pub title: String,
|
|
pub deadline: Option<chrono::NaiveDate>,
|
|
pub category: Category,
|
|
pub project_id: Option<i32>,
|
|
}
|
|
|
|
impl NewTask {
|
|
pub fn new(
|
|
title: String,
|
|
deadline: Option<chrono::NaiveDate>,
|
|
category: Category,
|
|
project_id: Option<i32>,
|
|
) -> Self {
|
|
Self {
|
|
title,
|
|
deadline,
|
|
category,
|
|
project_id,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<Task> for NewTask {
|
|
fn from(task: Task) -> Self {
|
|
Self::new(task.title, task.deadline, task.category, task.project_id)
|
|
}
|
|
}
|