cdb/src/transaction.rs

205 lines
6 KiB
Rust
Raw Normal View History

2024-09-12 08:58:49 +00:00
use futures::StreamExt;
2024-07-24 14:38:56 +00:00
use serde::{Deserialize, Serialize};
2024-06-22 00:05:22 +00:00
use serde_json::json;
2024-10-07 18:53:58 +00:00
use sqlx::prelude::FromRow;
2024-06-22 00:05:22 +00:00
2024-10-07 18:53:58 +00:00
use crate::{get_itemdb, get_locations, get_pg, routes::ToAPI};
2024-08-30 12:13:56 +00:00
2024-09-25 06:38:12 +00:00
// todo : produced / consumed by flow field?
2024-08-28 07:14:33 +00:00
/// A Transaction of an Item Variant
2024-10-07 18:53:58 +00:00
#[derive(Debug, Clone, Serialize, Deserialize, FromRow)]
2024-07-24 14:38:56 +00:00
pub struct Transaction {
2024-08-28 07:14:33 +00:00
/// UUID
2024-10-07 18:53:58 +00:00
pub id: uuid::Uuid,
2024-08-28 07:14:33 +00:00
/// Associated Item
2024-07-24 14:38:56 +00:00
pub item: String,
2024-08-28 07:14:33 +00:00
/// Associated Variant
2024-07-24 14:38:56 +00:00
pub variant: String,
2024-08-28 07:14:33 +00:00
/// Price of obtaining the Item
2024-10-07 18:53:58 +00:00
pub price: f64,
2024-08-28 07:14:33 +00:00
/// Origin of the Item
2024-07-24 14:38:56 +00:00
pub origin: Option<String>,
2024-09-02 16:40:02 +00:00
/// The location of the Item
2024-10-07 18:53:58 +00:00
pub location: Option<String>,
2024-09-20 23:38:22 +00:00
/// Notes on Transaction
pub note: Option<String>,
2024-08-28 07:14:33 +00:00
/// Timestamp of the Transaction
2024-10-07 18:53:58 +00:00
pub created: chrono::DateTime<chrono::Utc>,
2024-08-28 07:14:33 +00:00
/// Destination of the Item or who consumed it
2024-10-07 18:53:58 +00:00
pub destination: Option<String>,
2024-08-28 07:14:33 +00:00
/// Price the Item was exported or consumed at
2024-10-07 18:53:58 +00:00
pub consumed_price: Option<f64>,
2024-08-28 07:14:33 +00:00
/// Timestamp of Consumption
2024-10-07 18:53:58 +00:00
pub consumed_timestamp: Option<chrono::DateTime<chrono::Utc>>,
2024-05-03 16:22:59 +00:00
}
impl Transaction {
2024-09-02 16:40:02 +00:00
pub async fn new(
item: &str,
variant: &str,
2024-10-07 18:53:58 +00:00
price: f64,
2024-09-02 16:40:02 +00:00
origin: Option<&str>,
location: Option<&str>,
2024-09-20 23:38:22 +00:00
note: Option<&str>,
2024-09-02 16:40:02 +00:00
) -> Self {
2024-10-07 18:53:58 +00:00
sqlx::query_as("INSERT INTO transactions (item, variant, price, origin, location, note) VALUES ($1, $2, $3, $4, $5, $6) RETURNING *")
.bind(item)
.bind(variant)
.bind(price)
.bind(origin)
.bind(location)
.bind(note)
.fetch_one(get_pg!()).await.unwrap()
2024-05-03 16:22:59 +00:00
}
2024-10-07 18:53:58 +00:00
pub async fn get(id: &uuid::Uuid) -> Option<Self> {
sqlx::query_as("SELECT * FROM transactions WHERE id = $1")
.bind(id)
.fetch_optional(get_pg!())
2024-09-12 08:17:14 +00:00
.await
2024-09-12 09:48:09 +00:00
.unwrap()
2024-08-28 07:14:33 +00:00
}
2024-10-07 18:53:58 +00:00
/// Consumes the Item with `price` and `destination`
pub async fn consume(self, price: f64, destination: &str) -> Self {
sqlx::query_as(
"UPDATE transactions SET consumed_timestamp = current_timestamp, consumed_price = $1, destination = $2 WHERE id = $3 RETURNING *")
.bind(price)
.bind(destination)
.bind(&self.id)
.fetch_one(get_pg!()).await.unwrap()
}
2024-09-26 13:55:48 +00:00
pub async fn is_expired_at(&self, time: i64) -> bool {
2024-10-07 18:53:58 +00:00
if let Some(expiry) = get_itemdb!()
.get_item(&self.item)
2024-09-27 07:19:32 +00:00
.unwrap()
.variant(&self.variant)
.unwrap()
.expiry
{
2024-10-07 18:53:58 +00:00
let date_added = self.created.timestamp();
2024-08-30 12:13:56 +00:00
2024-09-27 07:19:32 +00:00
let expiration_ts = expiry * 24 * 60 * 60;
2024-08-30 12:13:56 +00:00
2024-09-27 07:19:32 +00:00
return (date_added + expiration_ts) < time;
}
2024-08-30 12:13:56 +00:00
2024-09-27 07:19:32 +00:00
false
2024-09-26 13:55:48 +00:00
}
pub async fn is_expired_in_days(&self, days: i64) -> bool {
let current_time = chrono::Utc::now().timestamp();
2024-09-27 07:19:32 +00:00
self.is_expired_at(current_time + (days * 24 * 60 * 60))
.await
2024-09-26 13:55:48 +00:00
}
pub async fn is_expired(&self) -> bool {
2024-10-07 18:53:58 +00:00
if self.consumed_timestamp.is_some() {
if let Some(expiry) = get_itemdb!()
.get_item(&self.item)
2024-09-27 07:19:32 +00:00
.unwrap()
.variant(&self.variant)
.unwrap()
.expiry
{
2024-10-07 18:53:58 +00:00
let time_around =
self.created.timestamp() - self.consumed_timestamp.unwrap().timestamp();
2024-09-27 07:19:32 +00:00
let expiration_ts = expiry * 24 * 60 * 60;
return time_around > expiration_ts;
} else {
return false;
}
}
2024-09-26 13:55:48 +00:00
let current_time = chrono::Utc::now().timestamp();
self.is_expired_at(current_time).await
2024-08-30 12:13:56 +00:00
}
2024-09-12 08:17:14 +00:00
2024-10-07 18:53:58 +00:00
pub async fn in_location(l: &str) -> Vec<Self> {
sqlx::query_as(
"SELECT * FROM transactions WHERE location = $1 AND consumed_timestamp IS NULL",
2024-09-22 00:38:03 +00:00
)
2024-10-07 18:53:58 +00:00
.bind(l)
.fetch_all(get_pg!())
.await
.unwrap()
2024-09-12 08:17:14 +00:00
}
2024-09-21 17:18:08 +00:00
pub async fn in_location_recursive(l: &str) -> Option<Vec<Self>> {
2024-09-22 00:38:03 +00:00
// get the children of this location
2024-10-07 18:53:58 +00:00
let locations = get_locations!().get(l)?.children_recursive();
2024-09-21 17:18:08 +00:00
2024-10-07 18:53:58 +00:00
let mut transactions = Self::in_location(l).await;
2024-09-12 08:34:14 +00:00
for loc in locations {
2024-10-07 18:53:58 +00:00
transactions.extend(Self::in_location(&loc.id).await);
2024-09-12 08:34:14 +00:00
}
2024-09-21 17:18:08 +00:00
Some(transactions)
2024-09-12 08:17:14 +00:00
}
2024-09-12 08:58:49 +00:00
/// Get all Transactions which are not consumed and are expired
2024-09-26 13:55:48 +00:00
pub async fn active_expired(days: Option<i64>) -> Vec<Self> {
2024-10-07 18:53:58 +00:00
let items: Vec<Self> = sqlx::query_as(
"SELECT * FROM transactions WHERE consumed_timestamp IS NULL ORDER BY created DESC",
2024-09-12 08:58:49 +00:00
)
2024-10-07 18:53:58 +00:00
.fetch_all(get_pg!())
2024-09-12 08:58:49 +00:00
.await
.unwrap();
let expired_items: Vec<_> = futures::stream::iter(items)
.filter_map(|item| async move {
2024-09-26 13:55:48 +00:00
let expired = if let Some(days) = days {
item.is_expired_in_days(days).await
} else {
item.is_expired().await
};
if expired {
2024-09-12 08:58:49 +00:00
Some(item)
} else {
None
}
})
.collect()
.await;
expired_items
}
2024-08-30 12:13:56 +00:00
}
2024-10-07 18:53:58 +00:00
impl ToAPI for Transaction {
2024-08-30 12:13:56 +00:00
async fn api(&self) -> serde_json::Value {
2024-09-23 18:07:08 +00:00
let location = if let Some(loc) = &self.location {
2024-10-07 18:53:58 +00:00
Some(get_locations!().get(loc).unwrap().api().await)
} else {
None
};
let consumed = if self.consumed_timestamp.is_some() {
Some(json!({
"destination": self.destination,
"price": self.consumed_price,
"timestamp": self.consumed_timestamp.unwrap().timestamp()
}))
2024-09-23 18:07:08 +00:00
} else {
None
};
2024-06-22 00:05:22 +00:00
json!({
2024-10-07 18:53:58 +00:00
"uuid": self.id,
2024-06-22 00:05:22 +00:00
"item": self.item,
"variant": self.variant,
"price": self.price,
"origin": self.origin,
2024-09-23 18:07:08 +00:00
"location": location,
2024-10-07 18:53:58 +00:00
"timestamp": self.created.timestamp(),
"consumed": consumed,
2024-09-20 23:38:22 +00:00
"note": self.note,
2024-08-30 12:13:56 +00:00
"expired": self.is_expired().await
2024-06-22 00:05:22 +00:00
})
}
2024-06-21 19:14:45 +00:00
}