2024-05-03 16:22:59 +00:00
|
|
|
use mongodb::bson::doc;
|
2024-06-22 00:05:22 +00:00
|
|
|
use serde_json::json;
|
2024-05-03 16:22:59 +00:00
|
|
|
|
|
|
|
use crate::{
|
2024-05-10 08:56:07 +00:00
|
|
|
cache::InventoryCache,
|
2024-06-22 00:05:22 +00:00
|
|
|
cdb_col, collect_results, get_mongo, id_of,
|
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-22 00:05:22 +00:00
|
|
|
supply_t.find_one(id_of!(uuid), None).await.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
|
|
|
|
2024-06-22 00:05:22 +00:00
|
|
|
pub fn api_json(&self) -> serde_json::Value {
|
|
|
|
json!({
|
|
|
|
"item": self.item,
|
|
|
|
"variant": self.variant,
|
|
|
|
"amount": self.amount as u32,
|
|
|
|
"depends": self.depends
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|