217 lines
6.7 KiB
Rust
217 lines
6.7 KiB
Rust
use axum::{http::StatusCode, response::IntoResponse};
|
|
use chrono::{DateTime, FixedOffset, Utc};
|
|
use rusqlite::{Connection, OptionalExtension};
|
|
use serde::Serialize;
|
|
use uuid::Uuid;
|
|
|
|
use crate::{database::DatabaseError, persons::Name};
|
|
|
|
#[derive(Serialize)]
|
|
pub struct Quote {
|
|
pub id: Uuid,
|
|
pub lines: Vec<QuoteLine>,
|
|
pub timestamp: DateTime<FixedOffset>,
|
|
pub location: Option<String>,
|
|
pub context: Option<String>,
|
|
pub created_by: Uuid,
|
|
pub public: bool,
|
|
}
|
|
|
|
#[derive(Serialize)]
|
|
pub struct QuoteLine {
|
|
pub id: Uuid,
|
|
pub attribution: Name,
|
|
pub content: String,
|
|
}
|
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
pub enum QuoteError {
|
|
#[error("No quote with ID {0}")]
|
|
NoQuoteWithId(Uuid),
|
|
#[error("A quote must have at least one line")]
|
|
EmptyQuote,
|
|
#[error("{0}")]
|
|
DatabaseError(#[from] DatabaseError),
|
|
}
|
|
|
|
impl Quote {
|
|
pub fn get_creation_timestamp(&self) -> DateTime<Utc> {
|
|
// unwrap here because all IDs use UUIDv7
|
|
let (s, n) = self.id.get_timestamp().unwrap().to_unix();
|
|
// unwrap here because timestamps held by UUIDs are valid by spec
|
|
DateTime::from_timestamp(s as i64, n).unwrap()
|
|
}
|
|
}
|
|
|
|
impl Quote {
|
|
pub fn total_count(conn: &Connection) -> Result<i64, QuoteError> {
|
|
Ok(conn.query_row("SELECT COUNT(*) FROM quotes", (), |r| r.get(0))?)
|
|
}
|
|
pub fn get_by_id(conn: &Connection, id: Uuid) -> Result<Quote, QuoteError> {
|
|
let quotemain = conn
|
|
.prepare(
|
|
"SELECT timestamp, location, context, created_by, public FROM quotes WHERE id = ?1",
|
|
)?
|
|
.query_row((id,), |r| {
|
|
Ok((
|
|
r.get::<_, DateTime<FixedOffset>>(0)?,
|
|
r.get::<_, Option<String>>(1)?,
|
|
r.get::<_, Option<String>>(2)?,
|
|
r.get::<_, Uuid>(3)?,
|
|
r.get::<_, bool>(4)?,
|
|
))
|
|
})
|
|
.optional()?;
|
|
|
|
let (timestamp, location, context, created_by, public) = match quotemain {
|
|
Some(data) => data,
|
|
None => return Err(QuoteError::NoQuoteWithId(id)),
|
|
};
|
|
|
|
let lines = conn
|
|
.prepare(
|
|
r#"
|
|
SELECT l.id, l.content, n.id, n.is_primary, n.person_id, n.created_by, n.name
|
|
FROM lines AS l JOIN names AS n ON l.name_id = n.id
|
|
WHERE l.quote_id = ?1 ORDER BY l.ordering
|
|
"#,
|
|
)?
|
|
.query_map((id,), |r| {
|
|
Ok(QuoteLine {
|
|
id: r.get(0)?,
|
|
content: r.get(1)?,
|
|
attribution: Name {
|
|
id: r.get(2)?,
|
|
is_primary: r.get(3)?,
|
|
person_id: r.get(4)?,
|
|
created_by: r.get(5)?,
|
|
name: r.get(6)?,
|
|
},
|
|
})
|
|
})?
|
|
.collect::<Result<Vec<QuoteLine>, _>>()?;
|
|
|
|
Ok(Quote {
|
|
id,
|
|
lines,
|
|
timestamp,
|
|
location,
|
|
context,
|
|
created_by,
|
|
public,
|
|
})
|
|
}
|
|
pub fn get_newest(conn: &Connection) -> Result<Option<Quote>, QuoteError> {
|
|
let id: Option<Uuid> = conn
|
|
.query_row("SELECT id FROM quotes ORDER BY id DESC LIMIT 1", (), |r| {
|
|
r.get(0)
|
|
})
|
|
.optional()?;
|
|
|
|
match id {
|
|
Some(id) => Ok(Some(Self::get_by_id(conn, id)?)),
|
|
None => Ok(None),
|
|
}
|
|
}
|
|
pub fn get_chronological_cursorscroll(
|
|
conn: &Connection,
|
|
cursor: Option<Uuid>,
|
|
amount: i64,
|
|
) -> Result<Vec<Quote>, QuoteError> {
|
|
let ids = match cursor {
|
|
Some(c) => conn
|
|
.prepare("SELECT id FROM quotes WHERE id < ?1 ORDER BY id DESC LIMIT ?2")?
|
|
.query_map((c, amount), |r| r.get(0))?
|
|
.collect::<Result<Vec<Uuid>, _>>()?,
|
|
None => conn
|
|
.prepare("SELECT id FROM quotes ORDER BY id DESC LIMIT ?1")?
|
|
.query_map((amount,), |r| r.get(0))?
|
|
.collect::<Result<Vec<Uuid>, _>>()?,
|
|
};
|
|
|
|
ids.iter().map(|id| Self::get_by_id(conn, *id)).collect()
|
|
}
|
|
pub fn get_chronological_offset(
|
|
conn: &Connection,
|
|
offset: i64,
|
|
limit: i64,
|
|
) -> Result<Vec<Quote>, QuoteError> {
|
|
let ids = conn
|
|
.prepare("SELECT id FROM quotes ORDER BY id DESC LIMIT ?1 OFFSET ?2")?
|
|
.query_map((limit, offset), |r| r.get(0))?
|
|
.collect::<Result<Vec<Uuid>, _>>()?;
|
|
|
|
ids.iter().map(|id| Self::get_by_id(conn, *id)).collect()
|
|
}
|
|
pub fn create(
|
|
conn: &Connection,
|
|
lines: Vec<(String, Name)>,
|
|
timestamp: DateTime<FixedOffset>,
|
|
context: Option<String>,
|
|
location: Option<String>,
|
|
created_by: Uuid,
|
|
public: bool,
|
|
) -> Result<Quote, QuoteError> {
|
|
if lines.is_empty() {
|
|
return Err(QuoteError::EmptyQuote);
|
|
}
|
|
|
|
let quote_id = Uuid::now_v7();
|
|
let lines: Vec<(Uuid, String, Name)> = lines
|
|
.into_iter()
|
|
.map(|(c, a)| (Uuid::now_v7(), c, a))
|
|
.collect();
|
|
|
|
let mut quote_stmt = conn.prepare(
|
|
r#"
|
|
INSERT INTO quotes (id, timestamp, location, context, created_by, public)
|
|
VALUES (?1, ?2, ?3, ?4, ?5, ?6)
|
|
"#,
|
|
)?;
|
|
quote_stmt.execute((quote_id, timestamp, &location, &context, created_by, public))?;
|
|
|
|
let mut line_stmt = conn.prepare(
|
|
r#"
|
|
INSERT INTO lines (id, quote_id, content, name_id, ordering)
|
|
VALUES (?1, ?2, ?3, ?4, ?5)
|
|
"#,
|
|
)?;
|
|
for (ordering, (id, content, attr)) in lines.iter().enumerate() {
|
|
line_stmt.execute((id, quote_id, content, attr.id, ordering as i64))?;
|
|
}
|
|
|
|
Ok(Quote {
|
|
id: quote_id,
|
|
lines: lines
|
|
.into_iter()
|
|
.map(|(id, content, attribution)| QuoteLine {
|
|
id,
|
|
content,
|
|
attribution,
|
|
})
|
|
.collect(),
|
|
timestamp,
|
|
location,
|
|
context,
|
|
created_by,
|
|
public,
|
|
})
|
|
}
|
|
}
|
|
|
|
impl From<rusqlite::Error> for QuoteError {
|
|
fn from(error: rusqlite::Error) -> Self {
|
|
QuoteError::DatabaseError(DatabaseError::from(error))
|
|
}
|
|
}
|
|
|
|
impl IntoResponse for QuoteError {
|
|
fn into_response(self) -> axum::response::Response {
|
|
match self {
|
|
Self::DatabaseError(e) => e.into_response(),
|
|
Self::NoQuoteWithId(_) => (StatusCode::BAD_REQUEST, self.to_string()).into_response(),
|
|
Self::EmptyQuote => (StatusCode::BAD_REQUEST, self.to_string()).into_response(),
|
|
}
|
|
}
|
|
}
|