cdb/src/variant.rs

195 lines
5.4 KiB
Rust
Raw Normal View History

2024-05-03 16:22:59 +00:00
use mongodb::bson::doc;
use crate::{
2024-05-10 08:56:07 +00:00
cache::InventoryCache,
2024-05-10 09:59:05 +00:00
cdb_col, collect_results, get_mongo,
2024-05-03 16:22:59 +00:00
transaction::{BatchTransaction, Price, Transaction},
};
/// Represents a specific instance of an item with potential variations.
///
/// This struct is used to describe a particular variation or instance of an item
/// in the real world. It may include attributes or properties that deviate from
/// the standard definition of the item. For example, different colors, sizes, or
/// configurations.
2024-05-06 06:24:23 +00:00
#[derive(Debug, Clone)]
2024-05-03 16:22:59 +00:00
pub struct Variant {
2024-05-10 09:59:05 +00:00
/// Associated Item
2024-05-03 16:22:59 +00:00
pub item: String,
2024-05-10 09:59:05 +00:00
/// Variant Name
2024-05-03 16:22:59 +00:00
pub variant: String,
2024-05-10 09:59:05 +00:00
/// Amount of items this variant represents
2024-05-06 06:24:23 +00:00
pub amount: u64,
2024-05-10 09:59:05 +00:00
/// Dependencies for the item variant.
2024-05-10 08:56:07 +00:00
pub depends: Vec<String>,
2024-05-03 16:22:59 +00:00
}
impl Variant {
2024-05-10 09:59:05 +00:00
/// Create variant from itemdb yaml
2024-05-06 06:24:23 +00:00
pub fn from_yml(json: &serde_yaml::Value, variant: &str, item: &str) -> Self {
Self {
item: item.to_string(),
variant: variant.to_string(),
amount: json
.as_mapping()
.unwrap()
.get("amount")
2024-06-21 19:14:45 +00:00
.map_or(1, |x| x.as_u64().unwrap()),
2024-05-10 08:56:07 +00:00
depends: json
.as_mapping()
.unwrap()
.get("depends")
.map(|x| {
x.as_sequence()
.unwrap()
2024-06-21 19:14:45 +00:00
.iter()
2024-05-10 08:56:07 +00:00
.map(|x| x.as_str().unwrap().to_string())
.collect()
})
2024-06-21 19:14:45 +00:00
.unwrap_or_default(),
2024-05-06 06:24:23 +00:00
}
}
2024-05-10 09:59:05 +00:00
pub async fn supply_log(&self) -> Vec<String> {
let db = get_mongo!();
let supply = cdb_col!(db, "supply");
let filter = doc! {
"item": &self.item,
"variant": &self.variant
};
let mut db_res = supply.find(filter, None).await.unwrap();
let result: Vec<mongodb::bson::Document> = collect_results!(db_res);
let mut ret = Vec::new();
for doc in result {
ret.push(doc.get("_id").unwrap().as_str().unwrap().to_string());
}
ret
}
pub async fn demand_log(&self) -> Vec<String> {
// todo : add referenced supply
let db = get_mongo!();
let demand = cdb_col!(db, "demand");
let filter = doc! {
"item": &self.item,
"variant": &self.variant
};
let mut db_res = demand.find(filter, None).await.unwrap();
let result: Vec<mongodb::bson::Document> = collect_results!(db_res);
let mut ret = Vec::new();
for doc in result {
ret.push(doc.get("_id").unwrap().as_str().unwrap().to_string());
}
ret
}
pub async fn demand(uuid: &str, price: Price, destination: &str) -> Option<String> {
2024-05-03 16:22:59 +00:00
let db = get_mongo!();
// check if supply transaction exists
let supply_t = cdb_col!(db, "supply");
2024-06-21 19:14:45 +00:00
supply_t
.find_one(doc! { "_id": uuid}, None)
2024-05-03 16:22:59 +00:00
.await
2024-06-21 19:14:45 +00:00
.unwrap()?;
2024-05-03 16:22:59 +00:00
// todo : demand batch
// mark as used
let demand_t = cdb_col!(db, "demand");
demand_t
.insert_one(
doc! {
"_id": uuid,
"destination": destination,
2024-06-21 19:14:45 +00:00
"price": Into::<mongodb::bson::Bson>::into(price)
2024-05-03 16:22:59 +00:00
},
None,
)
.await
.unwrap();
2024-05-10 09:59:05 +00:00
// update cache
InventoryCache::remove(uuid).await;
2024-05-03 16:22:59 +00:00
Some(uuid.to_string())
}
/// Records a supply transaction in the database.
///
/// # Arguments
///
/// * `amount` - The quantity of items supplied.
/// * `price` - The price of the supplied items.
/// * `origin` - The origin or source of the supplied items.
///
/// # Returns
///
/// Returns a UUID string representing the transaction.
pub async fn supply(&self, amount: usize, price: Price, origin: &str) -> String {
let db = get_mongo!();
let col: mongodb::Collection<mongodb::bson::Document> =
2024-05-10 09:03:34 +00:00
db.database("cdb").collection("supply");
2024-05-03 16:22:59 +00:00
let mut transactions = vec![];
for _ in 0..amount {
transactions.push(Transaction::new(
&self.item,
&self.variant,
price.clone(),
origin,
));
}
for transaction in &transactions {
2024-06-21 19:14:45 +00:00
col.insert_one(transaction.as_doc(), None).await.unwrap();
2024-05-10 08:56:07 +00:00
// update cache
2024-05-10 09:03:34 +00:00
InventoryCache::push(&transaction.uuid).await;
2024-05-03 16:22:59 +00:00
}
// batch transaction
let ret_uuid = if amount == 1 {
transactions.first().unwrap().uuid.clone()
} else {
let batch =
BatchTransaction::new(transactions.iter().map(|x| x.uuid.clone()).collect());
2024-05-10 09:03:34 +00:00
let col: mongodb::Collection<mongodb::bson::Document> =
db.database("cdb").collection("transactions_batch");
2024-06-21 19:14:45 +00:00
let batch_uuid = batch.uuid.clone();
col.insert_one(Into::<mongodb::bson::Document>::into(batch), None)
.await
.unwrap();
batch_uuid
2024-05-03 16:22:59 +00:00
};
ret_uuid
}
2024-05-10 08:56:07 +00:00
pub fn as_bson(&self) -> mongodb::bson::Document {
mongodb::bson::doc! {
"item": &self.item,
"variant": &self.variant,
"amount": self.amount as u32,
"depends": &self.depends
}
}
2024-05-03 16:22:59 +00:00
}