postgres
This commit is contained in:
parent
584ffb6b11
commit
1faa3b9668
19 changed files with 1058 additions and 1244 deletions
1149
Cargo.lock
generated
1149
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -4,11 +4,9 @@ version = "0.1.0"
|
|||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
chrono = "0.4.38"
|
||||
futures = "0.3.30"
|
||||
log = "0.4.20"
|
||||
mdq = { git = "https://git.hydrar.de/mdtools/mdq" }
|
||||
mongodb = "2.8.0"
|
||||
rocket = { version = "0.5.1", features = ["json"] }
|
||||
rocket_cors = "0.6.0"
|
||||
serde = { version = "1.0.195", features = ["derive"] }
|
||||
|
@ -16,8 +14,9 @@ serde_json = "1.0.111"
|
|||
serde_yaml = "0.9.34"
|
||||
tokio = { version = "1.35.1", features = ["full"] }
|
||||
toml = "0.8.8"
|
||||
uuid = { version = "1.8.0", features = ["v4"] }
|
||||
mongod = { git = "https://git.hydrar.de/jmarya/mongod" }
|
||||
env_logger = "0.11.5"
|
||||
walkdir = "2.5.0"
|
||||
reqwest = { version = "0.11", features = ["json"] }
|
||||
chrono = { version = "0.4.38", features = ["serde"] }
|
||||
uuid = { version = "1.8.0", features = ["v4", "serde"] }
|
||||
sqlx = { version = "0.8", features = ["postgres", "runtime-tokio-native-tls", "derive", "uuid", "chrono", "json"] }
|
||||
|
|
|
@ -5,25 +5,26 @@ services:
|
|||
ports:
|
||||
- "8080:8080"
|
||||
depends_on:
|
||||
- mongodb
|
||||
- postgres
|
||||
volumes:
|
||||
- ./itemdb/items:/itemdb
|
||||
- ./locations:/locations
|
||||
- ./flows:/flows
|
||||
- ./config.toml:/config.toml
|
||||
environment:
|
||||
- "DB_URI=mongodb://user:pass@mongodb:27017"
|
||||
- "DB=cdb"
|
||||
- "DATABASE_URL=postgres://user:pass@postgres/cdb"
|
||||
- "RUST_LOG=debug"
|
||||
- "ROCKET_ADDRESS=0.0.0.0"
|
||||
- "ROCKET_PORT=8080"
|
||||
|
||||
mongodb:
|
||||
image: mongo:latest
|
||||
postgres:
|
||||
image: timescale/timescaledb:latest-pg16
|
||||
restart: always
|
||||
ports:
|
||||
- "27017:27017"
|
||||
environment:
|
||||
MONGO_INITDB_ROOT_USERNAME: user
|
||||
MONGO_INITDB_ROOT_PASSWORD: pass
|
||||
- 5432:5432
|
||||
volumes:
|
||||
- ./db:/data/db
|
||||
- ./db:/var/lib/postgresql/data/
|
||||
environment:
|
||||
- POSTGRES_USER=user
|
||||
- POSTGRES_PASSWORD=pass
|
||||
- POSTGRES_DB=cdb
|
||||
|
|
32
migrations/0000_init.sql
Normal file
32
migrations/0000_init.sql
Normal file
|
@ -0,0 +1,32 @@
|
|||
CREATE TABLE flows (
|
||||
id UUID NOT NULL PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
"started" timestamptz NOT NULL DEFAULT current_timestamp,
|
||||
kind TEXT NOT NULL,
|
||||
input UUID[],
|
||||
ended timestamptz,
|
||||
"next" UUID,
|
||||
produced UUID[],
|
||||
FOREIGN KEY "next" REFERENCES flows(id)
|
||||
);
|
||||
|
||||
CREATE TABLE flow_notes (
|
||||
id UUID NOT NULL PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
time timestamptz NOT NULL DEFAULT current_timestamp,
|
||||
content TEXT NOT NULL,
|
||||
on_flow UUID NOT NULL,
|
||||
FOREIGN KEY on_flow REFERENCES flows(id)
|
||||
);
|
||||
|
||||
CREATE TABLE transactions (
|
||||
id UUID NOT NULL PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||
created timestamptz NOT NULL DEFAULT current_timestamp,
|
||||
item TEXT NOT NULL,
|
||||
variant TEXT NOT NULL,
|
||||
price NUMERIC(2),
|
||||
origin TEXT,
|
||||
"location" TEXT,
|
||||
note TEXT,
|
||||
destination TEXT,
|
||||
consumed_price NUMERIC(2),
|
||||
consumed_timestamp timestamptz
|
||||
);
|
|
@ -1,10 +1,9 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use mongod::Model;
|
||||
|
||||
use crate::item::Item;
|
||||
|
||||
/// Item database
|
||||
#[derive(Debug)]
|
||||
pub struct ItemDB {
|
||||
index: HashMap<String, Item>,
|
||||
}
|
||||
|
@ -20,9 +19,7 @@ impl ItemDB {
|
|||
|
||||
for item in &index.documents {
|
||||
let item = Item::new(item);
|
||||
item.insert_overwrite().await.unwrap();
|
||||
log::info!("Adding item {} to DB", item.name);
|
||||
items.insert(item._id.clone(), item);
|
||||
items.insert(item.id.clone(), item);
|
||||
}
|
||||
|
||||
Self { index: items }
|
||||
|
|
245
src/flow.rs
245
src/flow.rs
|
@ -1,11 +1,16 @@
|
|||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
use std::collections::HashMap;
|
||||
|
||||
use mongod::{assert_reference_of, reference_of, Reference, Validate};
|
||||
use crate::routes::item::{item_does_not_exist_error, variant_does_not_exist_error, SupplyForm};
|
||||
use crate::routes::{ApiError, ToAPI};
|
||||
use crate::{get_itemdb, get_pg};
|
||||
use sqlx::FromRow;
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub struct FlowInfo {
|
||||
#[serde(default)]
|
||||
pub _id: String,
|
||||
pub id: String,
|
||||
pub name: String,
|
||||
#[serde(default)]
|
||||
pub depends: Vec<String>,
|
||||
|
@ -13,23 +18,10 @@ pub struct FlowInfo {
|
|||
pub produces: Option<Vec<String>>,
|
||||
}
|
||||
|
||||
impl Validate for FlowInfo {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl FlowInfo {
|
||||
pub async fn add(&mut self, id: &str) {
|
||||
self._id = id.to_string();
|
||||
self.insert_overwrite().await.unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
impl ToAPI for FlowInfo {
|
||||
async fn api(&self) -> serde_json::Value {
|
||||
json!({
|
||||
"id": self._id,
|
||||
"id": self.id,
|
||||
"name": self.name,
|
||||
"depends": self.depends,
|
||||
"next": self.next,
|
||||
|
@ -37,119 +29,68 @@ impl ToAPI for FlowInfo {
|
|||
})
|
||||
}
|
||||
}
|
||||
|
||||
use mongod::{
|
||||
derive::{Model, Referencable},
|
||||
Model, Referencable, ToAPI,
|
||||
};
|
||||
use mongodb::bson::doc;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
|
||||
use crate::item::Item;
|
||||
use crate::routes::item::{item_does_not_exist_error, variant_does_not_exist_error, SupplyForm};
|
||||
use crate::routes::{api_error, ApiError};
|
||||
use crate::transaction::{Price, Transaction};
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
|
||||
pub struct DoneInfo {
|
||||
/// Timestamp when the flow was ended
|
||||
pub ended: i64,
|
||||
/// The flow succedding this one
|
||||
pub next: Option<Reference>,
|
||||
/// Transactions this flow produced
|
||||
pub produced: Option<Vec<Reference>>,
|
||||
}
|
||||
|
||||
impl DoneInfo {
|
||||
pub fn new(next: Option<Reference>) -> Self {
|
||||
Self {
|
||||
ended: chrono::Utc::now().timestamp(),
|
||||
next,
|
||||
produced: None,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn api(&self) -> serde_json::Value {
|
||||
json!({
|
||||
"ended": self.ended,
|
||||
"next": self.next.as_ref().map(|x| x.id()),
|
||||
"produced": self.produced.as_ref().map(|x| x.iter().map(|t| t.id()).collect::<Vec<_>>()),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/// A production flow
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, FromRow)]
|
||||
pub struct Flow {
|
||||
/// ID
|
||||
pub _id: String,
|
||||
pub id: uuid::Uuid,
|
||||
/// Tiemstamp when the flow was started
|
||||
pub started: i64,
|
||||
pub started: chrono::DateTime<chrono::Utc>,
|
||||
/// Kind of flow; ID of the describing JSON
|
||||
pub kind: Reference,
|
||||
pub kind: String,
|
||||
/// Input transactions
|
||||
pub input: Option<Vec<Reference>>,
|
||||
/// Information when a flow is done
|
||||
pub done: Option<DoneInfo>,
|
||||
}
|
||||
|
||||
impl Validate for Flow {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
assert_reference_of!(self.kind, FlowInfo);
|
||||
|
||||
if let Some(input) = &self.input {
|
||||
for t in input {
|
||||
assert_reference_of!(t, Transaction);
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(done) = &self.done {
|
||||
if let Some(next) = &done.next {
|
||||
assert_reference_of!(next, Flow);
|
||||
}
|
||||
if let Some(produced) = &done.produced {
|
||||
for prod in produced {
|
||||
assert_reference_of!(prod, Transaction);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
pub input: Option<Vec<uuid::Uuid>>,
|
||||
/// Timestamp when the flow was ended
|
||||
pub ended: Option<chrono::DateTime<chrono::Utc>>,
|
||||
/// The flow succedding this one
|
||||
pub next: Option<uuid::Uuid>,
|
||||
/// Transactions this flow produced
|
||||
pub produced: Option<Vec<uuid::Uuid>>,
|
||||
}
|
||||
|
||||
impl ToAPI for Flow {
|
||||
async fn api(&self) -> serde_json::Value {
|
||||
let done = if self.ended.is_some() {
|
||||
Some(json!({
|
||||
"ended": self.ended.map(|x| x.timestamp()),
|
||||
"next": self.next,
|
||||
"produced": self.produced
|
||||
}))
|
||||
} else {
|
||||
None
|
||||
};
|
||||
|
||||
json!({
|
||||
"id": self._id,
|
||||
"started": self.started,
|
||||
"kind": self.kind.id(),
|
||||
"input": self.input.as_ref().map(|x| x.iter().map(|t| t.id()).collect::<Vec<_>>()),
|
||||
"done": self.done.as_ref().map(|x| x.api())
|
||||
"id": self.id,
|
||||
"started": self.started.timestamp(),
|
||||
"kind": self.kind,
|
||||
"input": self.input,
|
||||
"done": done
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl Flow {
|
||||
pub async fn create(kind: &str, input: Option<Vec<Reference>>) -> Self {
|
||||
let f = Self {
|
||||
_id: uuid::Uuid::new_v4().to_string(),
|
||||
started: chrono::Utc::now().timestamp(),
|
||||
kind: reference_of!(FlowInfo, kind).unwrap(),
|
||||
input: input,
|
||||
done: None,
|
||||
};
|
||||
pub async fn get(id: &uuid::Uuid) -> Option<Self> {
|
||||
sqlx::query_as("SELECT * FROM flows WHERE id = $1")
|
||||
.bind(id)
|
||||
.fetch_optional(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
f.insert().await.unwrap();
|
||||
|
||||
f
|
||||
pub async fn create(kind: &str, input: Option<Vec<uuid::Uuid>>) -> Self {
|
||||
sqlx::query_as("INSERT INTO flows (kind, input) VALUES ($1, $2) RETURNING *")
|
||||
.bind(kind)
|
||||
.bind(input)
|
||||
.fetch_one(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
pub async fn end(self) -> Self {
|
||||
self.change()
|
||||
.done(Some(DoneInfo::new(None)))
|
||||
.update()
|
||||
sqlx::query_as("UPDATE flows SET ended = current_timestamp RETURNING *")
|
||||
.fetch_one(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
@ -157,14 +98,14 @@ impl Flow {
|
|||
pub async fn end_with_produce(
|
||||
self,
|
||||
produced: &[SupplyForm],
|
||||
) -> Result<HashMap<String, Vec<String>>, ApiError> {
|
||||
) -> Result<HashMap<String, Vec<uuid::Uuid>>, ApiError> {
|
||||
let mut ret = HashMap::new();
|
||||
let mut t_create = Vec::new();
|
||||
let mut produced_ref = Vec::with_capacity(ret.len());
|
||||
|
||||
for prod in produced {
|
||||
let t = Item::get(&prod.item)
|
||||
.await
|
||||
let t = get_itemdb!()
|
||||
.get_item(&prod.item)
|
||||
.ok_or_else(item_does_not_exist_error)?
|
||||
.variant(&prod.variant)
|
||||
.ok_or_else(variant_does_not_exist_error)?;
|
||||
|
@ -175,83 +116,73 @@ impl Flow {
|
|||
for (item, info) in t_create {
|
||||
let t = item
|
||||
.supply(
|
||||
Price::zero(),
|
||||
Some(&format!("flow::{}::{}", self.kind.id(), self._id)),
|
||||
0.0,
|
||||
Some(&format!("flow::{}::{}", self.kind, self.id)),
|
||||
info.location.as_ref().map(|x| x.as_str()),
|
||||
info.note.as_ref().map(|x| x.as_str()),
|
||||
)
|
||||
.await;
|
||||
ret.entry(item.item_variant_id().clone())
|
||||
.or_insert(Vec::new())
|
||||
.push(t._id.clone());
|
||||
produced_ref.push(t.reference());
|
||||
.push(t.id.clone());
|
||||
produced_ref.push(t.id);
|
||||
}
|
||||
|
||||
self.change()
|
||||
.done(Some(DoneInfo {
|
||||
ended: chrono::Utc::now().timestamp(),
|
||||
next: None,
|
||||
produced: Some(produced_ref),
|
||||
}))
|
||||
.update()
|
||||
.await
|
||||
.unwrap();
|
||||
sqlx::query("UPDATE transactions SET consumed_timestamp = current_timestamp, produced = $1 WHERE id = $2")
|
||||
.bind(produced_ref)
|
||||
.bind(self.id)
|
||||
.execute(get_pg!()).await.unwrap();
|
||||
|
||||
Ok(ret)
|
||||
}
|
||||
|
||||
pub async fn continue_next(self, next_flow: &Flow) -> Self {
|
||||
self.change()
|
||||
.done(Some(DoneInfo::new(Some(next_flow.reference()))))
|
||||
.update()
|
||||
.await
|
||||
.unwrap()
|
||||
sqlx::query_as("UPDATE transactions SET consumed_timestamp = current_timestamp, \"next\" = $1 WHERE id = $2 RETURNING *")
|
||||
.bind(next_flow.id)
|
||||
.bind(&self.id)
|
||||
.fetch_one(get_pg!()).await.unwrap()
|
||||
}
|
||||
}
|
||||
|
||||
/// A note for a Flow
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, FromRow)]
|
||||
pub struct FlowNote {
|
||||
/// ID
|
||||
pub _id: String,
|
||||
/// Tiemstamp when the note was created
|
||||
pub timestamp: i64,
|
||||
pub id: uuid::Uuid,
|
||||
/// Timestamp when the note was created
|
||||
pub time: chrono::DateTime<chrono::Utc>,
|
||||
/// Note Content
|
||||
pub content: String,
|
||||
/// Associated flow
|
||||
pub on_flow: Reference,
|
||||
pub on_flow: uuid::Uuid,
|
||||
}
|
||||
|
||||
impl FlowNote {
|
||||
pub async fn create(content: &str, flow: Reference) -> Self {
|
||||
let s = Self {
|
||||
_id: uuid::Uuid::new_v4().to_string(),
|
||||
timestamp: chrono::Utc::now().timestamp(),
|
||||
content: content.to_string(),
|
||||
on_flow: flow,
|
||||
};
|
||||
pub async fn create(content: &str, flow: &uuid::Uuid) -> Self {
|
||||
sqlx::query_as("INSERT INTO flow_notes (content, on_flow) VALUES ($1, $2) RETURNING *")
|
||||
.bind(content)
|
||||
.bind(flow)
|
||||
.fetch_one(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
s.insert().await.unwrap();
|
||||
|
||||
s
|
||||
pub async fn find_of(id: &str) -> Vec<Self> {
|
||||
sqlx::query_as("SELECT * FROM flow_notes WHERE on_flow = $1 ORDER BY time DESC")
|
||||
.bind(id)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
}
|
||||
|
||||
impl ToAPI for FlowNote {
|
||||
async fn api(&self) -> serde_json::Value {
|
||||
json!({
|
||||
"uuid": self._id,
|
||||
"timestamp": self.timestamp,
|
||||
"uuid": self.id,
|
||||
"timestamp": self.time.timestamp(),
|
||||
"content": self.content,
|
||||
"on_flow": self.on_flow.id()
|
||||
"on_flow": self.on_flow
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl Validate for FlowNote {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
assert_reference_of!(self.on_flow, Flow);
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
|
52
src/item.rs
52
src/item.rs
|
@ -1,13 +1,9 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use mongod::{
|
||||
derive::{Model, Referencable},
|
||||
Model, Validate,
|
||||
};
|
||||
use mongodb::bson::doc;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
|
||||
use crate::get_pg;
|
||||
use crate::transaction::Transaction;
|
||||
use crate::variant::Variant;
|
||||
|
||||
|
@ -24,10 +20,10 @@ use crate::variant::Variant;
|
|||
/// This struct serves as a blueprint for describing individual items within
|
||||
/// a larger inventory or database of physical goods. It includes fields for
|
||||
/// the name of the item and its category.
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub struct Item {
|
||||
/// The ID
|
||||
pub _id: String,
|
||||
pub id: String,
|
||||
/// The name of the Item
|
||||
pub name: String,
|
||||
/// Category of the Item
|
||||
|
@ -37,12 +33,6 @@ pub struct Item {
|
|||
pub variants: HashMap<String, Variant>,
|
||||
}
|
||||
|
||||
impl Validate for Item {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Item {
|
||||
/// Creates a new `Item` from a parsed markdown document
|
||||
pub fn new(doc: &mdq::Document) -> Self {
|
||||
|
@ -93,7 +83,7 @@ impl Item {
|
|||
}
|
||||
|
||||
Self {
|
||||
_id: id,
|
||||
id,
|
||||
name,
|
||||
category,
|
||||
variants,
|
||||
|
@ -113,33 +103,23 @@ impl Item {
|
|||
}
|
||||
|
||||
pub async fn inventory(&self) -> Vec<Transaction> {
|
||||
let filter = doc! {
|
||||
"item": &self._id,
|
||||
"consumed": { "$not": { "$type": "object" } }
|
||||
};
|
||||
|
||||
Transaction::find(filter, None, None).await.unwrap()
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE item = $1 AND consumed_timestamp IS NULL ORDER BY created DESC")
|
||||
.bind(&self.id)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
}
|
||||
|
||||
pub async fn inventory_by_origin(&self, origin: &str) -> Vec<Transaction> {
|
||||
let filter = doc! {
|
||||
"item": &self._id,
|
||||
"consumed": { "$not": { "$type": "object" } },
|
||||
"origin": origin
|
||||
};
|
||||
|
||||
Transaction::find(filter, None, None).await.unwrap()
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE item = $1 AND consumed_timestamp IS NULL AND origin = $2 ORDER BY created DESC")
|
||||
.bind(&self.id)
|
||||
.bind(origin)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
}
|
||||
|
||||
pub async fn consumed_by_destination(&self, destination: &str) -> Vec<Transaction> {
|
||||
let filter = doc! {
|
||||
"item": &self._id,
|
||||
"consumed": {
|
||||
"destination": destination
|
||||
}
|
||||
};
|
||||
|
||||
Transaction::find(filter, None, None).await.unwrap()
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE item = $1 AND consumed_timestamp IS NOT NULL AND destination = $2 ORDER BY created DESC")
|
||||
.bind(&self.id)
|
||||
.bind(destination)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
}
|
||||
|
||||
pub fn api_json(&self) -> serde_json::Value {
|
||||
|
@ -150,7 +130,7 @@ impl Item {
|
|||
.collect();
|
||||
|
||||
json!({
|
||||
"uuid": self._id,
|
||||
"uuid": self.id,
|
||||
"name": self.name,
|
||||
"category": self.category,
|
||||
"variants": variants
|
||||
|
|
|
@ -5,6 +5,7 @@ use std::{
|
|||
|
||||
use serde::Deserialize;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct JSONStore<T> {
|
||||
documents: HashMap<String, T>,
|
||||
}
|
||||
|
|
|
@ -1,18 +1,14 @@
|
|||
use futures::FutureExt;
|
||||
use mongod::{
|
||||
derive::{Model, Referencable},
|
||||
Model, ToAPI, Validate,
|
||||
};
|
||||
use mongodb::bson::doc;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
|
||||
use crate::{get_locations, routes::ToAPI};
|
||||
|
||||
/// A Storage Location
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub struct Location {
|
||||
/// UUID
|
||||
#[serde(default)]
|
||||
pub _id: String,
|
||||
pub id: String,
|
||||
/// Name
|
||||
pub name: String,
|
||||
/// Parent
|
||||
|
@ -27,23 +23,18 @@ pub struct StorageConditions {
|
|||
pub temperature: i64,
|
||||
}
|
||||
|
||||
impl Validate for Location {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Location {
|
||||
/// Recursively get the conditions of a location. This inherits from parent locations.
|
||||
pub fn conditions_rec(&self) -> futures::future::BoxFuture<'_, Option<StorageConditions>> {
|
||||
async move {
|
||||
pub fn conditions_rec(&self) -> Option<StorageConditions> {
|
||||
let locations = get_locations!();
|
||||
|
||||
if let Some(cond) = &self.conditions {
|
||||
return Some(cond.clone());
|
||||
}
|
||||
|
||||
if let Some(parent) = &self.parent {
|
||||
if let Some(parent_loc) = Location::get(parent).await {
|
||||
if let Some(cond) = parent_loc.conditions_rec().await {
|
||||
if let Some(parent_loc) = locations.get(parent) {
|
||||
if let Some(cond) = parent_loc.conditions_rec() {
|
||||
return Some(cond);
|
||||
}
|
||||
}
|
||||
|
@ -51,49 +42,45 @@ impl Location {
|
|||
|
||||
None
|
||||
}
|
||||
.boxed()
|
||||
}
|
||||
|
||||
// Get direct children
|
||||
pub async fn children_direct(&self) -> Vec<Location> {
|
||||
Location::find(doc! { "parent": self._id.clone()}, None, None)
|
||||
.await
|
||||
.unwrap()
|
||||
pub fn children_direct(&self) -> Vec<Location> {
|
||||
let mut ret = Vec::new();
|
||||
|
||||
let locations = get_locations!();
|
||||
for loc in locations.keys() {
|
||||
let loc = locations.get(loc).unwrap();
|
||||
if *loc.parent.as_ref().unwrap_or(&String::new()) == self.id {
|
||||
ret.push(loc.clone());
|
||||
}
|
||||
}
|
||||
|
||||
ret
|
||||
}
|
||||
|
||||
// Get all children locations
|
||||
pub fn children_recursive(&self) -> futures::future::BoxFuture<'_, Vec<Location>> {
|
||||
async move {
|
||||
pub fn children_recursive(&self) -> Vec<Location> {
|
||||
let mut all = Vec::new();
|
||||
|
||||
let direct = self.children_direct().await;
|
||||
let direct = self.children_direct();
|
||||
all.extend_from_slice(&direct);
|
||||
|
||||
for loc in direct {
|
||||
let sub = loc.children_recursive().await;
|
||||
let sub = loc.children_recursive();
|
||||
all.extend_from_slice(&sub);
|
||||
}
|
||||
|
||||
all
|
||||
}
|
||||
.boxed()
|
||||
}
|
||||
}
|
||||
|
||||
impl ToAPI for Location {
|
||||
async fn api(&self) -> serde_json::Value {
|
||||
json!({
|
||||
"id": self._id,
|
||||
"id": self.id,
|
||||
"name": self.name,
|
||||
"parent": self.parent,
|
||||
"conditions": self.conditions_rec().await
|
||||
"conditions": self.conditions_rec()
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl Location {
|
||||
pub async fn add(&mut self, id: &str) {
|
||||
self._id = id.to_string();
|
||||
self.insert_overwrite().await.unwrap();
|
||||
}
|
||||
}
|
||||
|
|
64
src/main.rs
64
src/main.rs
|
@ -4,6 +4,7 @@ use location::Location;
|
|||
|
||||
use rocket::routes as route;
|
||||
use rocket::{http::Method, launch};
|
||||
use tokio::sync::OnceCell;
|
||||
|
||||
mod config;
|
||||
mod db;
|
||||
|
@ -16,6 +17,55 @@ mod routes;
|
|||
mod transaction;
|
||||
mod variant;
|
||||
|
||||
pub static PG: OnceCell<sqlx::PgPool> = OnceCell::const_new();
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_pg {
|
||||
() => {
|
||||
if let Some(client) = $crate::PG.get() {
|
||||
client
|
||||
} else {
|
||||
let client = sqlx::postgres::PgPoolOptions::new()
|
||||
.max_connections(5)
|
||||
.connect(&std::env::var("DATABASE_URL").unwrap())
|
||||
.await
|
||||
.unwrap();
|
||||
$crate::PG.set(client).unwrap();
|
||||
$crate::PG.get().unwrap()
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
pub static ITEMDB: OnceCell<db::ItemDB> = OnceCell::const_new();
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_itemdb {
|
||||
() => {
|
||||
if let Some(client) = $crate::ITEMDB.get() {
|
||||
client
|
||||
} else {
|
||||
let itemdb = $crate::db::ItemDB::new("./itemdb").await;
|
||||
$crate::ITEMDB.set(itemdb).unwrap();
|
||||
$crate::ITEMDB.get().unwrap()
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
pub static LOCATIONS: OnceCell<JSONStore<Location>> = OnceCell::const_new();
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! get_locations {
|
||||
() => {
|
||||
if let Some(client) = $crate::LOCATIONS.get() {
|
||||
client
|
||||
} else {
|
||||
let locations = $crate::JSONStore::new("./locations");
|
||||
$crate::LOCATIONS.set(locations).unwrap();
|
||||
$crate::LOCATIONS.get().unwrap()
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
// ░░░░░░░░░░▀▀▀██████▄▄▄░░░░░░░░░░
|
||||
// ░░░░░░░░░░░░░░░░░▀▀▀████▄░░░░░░░
|
||||
// ░░░░░░░░░░▄███████▀░░░▀███▄░░░░░
|
||||
|
@ -49,19 +99,11 @@ async fn rocket() -> _ {
|
|||
.expect("error creating CORS options");
|
||||
|
||||
let config = config::get_config();
|
||||
let itemdb = db::ItemDB::new("./itemdb").await;
|
||||
let mut locations: JSONStore<Location> = JSONStore::new("./locations");
|
||||
let mut flows: JSONStore<FlowInfo> = JSONStore::new("./flows");
|
||||
let itemdb = get_itemdb!();
|
||||
let locations = get_locations!();
|
||||
let flows: JSONStore<FlowInfo> = JSONStore::new("./flows");
|
||||
integrity::verify_integrity(&config, &flows, &locations, &itemdb).await;
|
||||
|
||||
for location in &mut *locations {
|
||||
location.1.add(location.0).await;
|
||||
}
|
||||
|
||||
for flow in &mut *flows {
|
||||
flow.1.add(flow.0).await;
|
||||
}
|
||||
|
||||
rocket::build()
|
||||
.mount(
|
||||
"/",
|
||||
|
|
|
@ -1,18 +1,16 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use mongod::{reference_of, vec_to_api, Model, Referencable, Sort, ToAPI};
|
||||
use mongodb::bson::doc;
|
||||
use rocket::{get, post, serde::json::Json, State};
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
use std::{collections::HashMap, str::FromStr};
|
||||
|
||||
use crate::{
|
||||
check_auth,
|
||||
config::Config,
|
||||
flow::{Flow, FlowInfo, FlowNote},
|
||||
get_pg,
|
||||
json_store::JSONStore,
|
||||
routes::{api_error, FallibleApiResponse, Token},
|
||||
transaction::{Price, Transaction},
|
||||
routes::{api_error, vec_to_api, FallibleApiResponse, ToAPI, Token},
|
||||
transaction::Transaction,
|
||||
};
|
||||
|
||||
use super::{item::SupplyForm, ApiError};
|
||||
|
@ -42,14 +40,9 @@ pub async fn active_flows_route(
|
|||
|
||||
let flowinfo = flows.get(id).ok_or_else(|| api_error("Flow not found"))?;
|
||||
|
||||
let flow = Flow::find(
|
||||
doc! {
|
||||
"kind": flowinfo.reference(),
|
||||
"done": { "$not": { "$type": "object" } }
|
||||
},
|
||||
None,
|
||||
None,
|
||||
)
|
||||
let flow: Vec<Flow> = sqlx::query_as("SELECT * FROM flows WHERE kind = $1 AND ended IS NULL")
|
||||
.bind(&flowinfo.id)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
|
@ -85,7 +78,8 @@ pub async fn create_flow(
|
|||
// verify valid input transactions
|
||||
if let Some(input) = &form.input {
|
||||
for t in input {
|
||||
let t = Transaction::get(t)
|
||||
let t =
|
||||
Transaction::get(&uuid::Uuid::from_str(t).map_err(|_| api_error("Invalid UUID"))?)
|
||||
.await
|
||||
.ok_or_else(|| api_error(&format!("No such transaction {t}")))?;
|
||||
|
||||
|
@ -109,14 +103,13 @@ pub async fn create_flow(
|
|||
if input_ref.is_empty() {
|
||||
None
|
||||
} else {
|
||||
Some(input_ref.iter().map(|x| x.reference()).collect())
|
||||
Some(input_ref.iter().map(|x| x.id).collect())
|
||||
},
|
||||
)
|
||||
.await;
|
||||
|
||||
for t in input_ref {
|
||||
t.consume(Price::zero(), &format!("flow::{kind}::{}", flow._id))
|
||||
.await;
|
||||
t.consume(0.0, &format!("flow::{kind}::{}", flow.id)).await;
|
||||
}
|
||||
|
||||
Ok(flow)
|
||||
|
@ -139,12 +132,12 @@ pub async fn create_flow_route(
|
|||
flows: &State<JSONStore<FlowInfo>>,
|
||||
) -> FallibleApiResponse {
|
||||
let flow = create_flow(id, flows, &form).await?;
|
||||
Ok(json!({"uuid": flow._id }))
|
||||
Ok(json!({"uuid": flow.id }))
|
||||
}
|
||||
|
||||
#[get("/flow/<id>")]
|
||||
pub async fn flow_api_route(id: &str) -> FallibleApiResponse {
|
||||
let flow = Flow::get(id)
|
||||
let flow = Flow::get(&uuid::Uuid::from_str(id).map_err(|_| api_error("Invalid UUID"))?)
|
||||
.await
|
||||
.ok_or_else(|| api_error("No such flow"))?;
|
||||
Ok(flow.api().await)
|
||||
|
@ -152,11 +145,11 @@ pub async fn flow_api_route(id: &str) -> FallibleApiResponse {
|
|||
|
||||
#[post("/flow/<id>/end", data = "<form>")]
|
||||
pub async fn end_flow_route(id: &str, form: Json<EndFlow>) -> FallibleApiResponse {
|
||||
let flow = Flow::get(id)
|
||||
let flow = Flow::get(&uuid::Uuid::from_str(id).map_err(|_| api_error("Invalid UUID"))?)
|
||||
.await
|
||||
.ok_or_else(|| api_error("Flow not found"))?;
|
||||
|
||||
if flow.done.is_some() {
|
||||
if flow.ended.is_some() {
|
||||
return Err(api_error("Flow already ended"));
|
||||
}
|
||||
|
||||
|
@ -175,17 +168,17 @@ pub async fn continue_flow_route(
|
|||
flows: &State<JSONStore<FlowInfo>>,
|
||||
form: Json<CreateFlow>,
|
||||
) -> FallibleApiResponse {
|
||||
let this_flow = Flow::get(id)
|
||||
let this_flow = Flow::get(&uuid::Uuid::from_str(id).map_err(|_| api_error("Invalid UUID"))?)
|
||||
.await
|
||||
.ok_or_else(|| api_error("Flow not found"))?;
|
||||
|
||||
if this_flow.done.is_some() {
|
||||
if this_flow.ended.is_some() {
|
||||
return Err(api_error("Flow already ended"));
|
||||
}
|
||||
|
||||
// create next flow
|
||||
let next_kind = flows
|
||||
.get(this_flow.kind.id())
|
||||
.get(&this_flow.kind)
|
||||
.ok_or_else(|| api_error("Flow not found"))?
|
||||
.next
|
||||
.clone()
|
||||
|
@ -195,20 +188,12 @@ pub async fn continue_flow_route(
|
|||
// end current flow
|
||||
this_flow.continue_next(&next_flow).await;
|
||||
|
||||
Ok(json!({"uuid": next_flow._id}))
|
||||
Ok(json!({"uuid": next_flow.id}))
|
||||
}
|
||||
|
||||
#[get("/flow/<id>/notes")]
|
||||
pub async fn flow_notes_route(id: &str) -> FallibleApiResponse {
|
||||
let notes = FlowNote::find(
|
||||
doc! {
|
||||
"on_flow": reference_of!(Flow, id).ok_or_else(|| api_error("No such flow"))?
|
||||
},
|
||||
None,
|
||||
Some(doc! { "timestamp": Sort::Descending }),
|
||||
)
|
||||
.await
|
||||
.unwrap();
|
||||
let notes = FlowNote::find_of(id).await;
|
||||
|
||||
Ok(json!(vec_to_api(¬es).await))
|
||||
}
|
||||
|
@ -222,8 +207,8 @@ pub struct NoteAdd {
|
|||
pub async fn create_flow_note_route(id: &str, form: Json<NoteAdd>) -> FallibleApiResponse {
|
||||
let note = FlowNote::create(
|
||||
&form.content,
|
||||
reference_of!(Flow, id).ok_or_else(|| api_error("No such flow"))?,
|
||||
&uuid::Uuid::from_str(id).map_err(|_| api_error("Invalid UUID"))?,
|
||||
)
|
||||
.await;
|
||||
Ok(json!({"uuid": note._id }))
|
||||
Ok(json!({"uuid": note.id }))
|
||||
}
|
||||
|
|
|
@ -1,14 +1,14 @@
|
|||
use mongod::{Model, ToAPI};
|
||||
use std::str::FromStr;
|
||||
|
||||
use rocket::serde::json::Json;
|
||||
use rocket::{get, post, State};
|
||||
use serde::Deserialize;
|
||||
use serde_json::json;
|
||||
|
||||
use crate::check_auth;
|
||||
use crate::config::{Config, Webhook};
|
||||
use crate::item::Item;
|
||||
use crate::routes::Token;
|
||||
use crate::routes::{ToAPI, Token};
|
||||
use crate::variant::Variant;
|
||||
use crate::{check_auth, get_itemdb};
|
||||
use crate::{
|
||||
db::ItemDB,
|
||||
routes::{api_error, FallibleApiResponse},
|
||||
|
@ -20,7 +20,7 @@ use super::{item_does_not_exist_error, variant_does_not_exist_error};
|
|||
pub struct DemandForm {
|
||||
uuid: String,
|
||||
destination: String,
|
||||
price: String,
|
||||
price: f64,
|
||||
}
|
||||
|
||||
/// Consumes a Transaction with Price and Destination
|
||||
|
@ -29,11 +29,8 @@ pub async fn demand_route(f: Json<DemandForm>, t: Token, c: &State<Config>) -> F
|
|||
check_auth!(t, c);
|
||||
|
||||
let transaction = Variant::demand(
|
||||
&f.uuid,
|
||||
f.price
|
||||
.clone()
|
||||
.try_into()
|
||||
.map_err(|()| api_error("Price malformed"))?,
|
||||
&uuid::Uuid::from_str(&f.uuid).map_err(|_| api_error("Invalid UUID"))?,
|
||||
f.price,
|
||||
&f.destination,
|
||||
)
|
||||
.await
|
||||
|
@ -45,8 +42,8 @@ pub async fn demand_route(f: Json<DemandForm>, t: Token, c: &State<Config>) -> F
|
|||
}
|
||||
|
||||
if let Some(url) = &hook.item_below_minimum {
|
||||
let variant = Item::get(&transaction.item)
|
||||
.await
|
||||
let variant = get_itemdb!()
|
||||
.get_item(&transaction.item)
|
||||
.unwrap()
|
||||
.variant(&transaction.variant)
|
||||
.unwrap();
|
||||
|
|
|
@ -1,6 +1,5 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use mongod::{vec_to_api, ToAPI};
|
||||
use rocket::{get, State};
|
||||
use serde_json::json;
|
||||
|
||||
|
@ -9,7 +8,7 @@ use crate::{
|
|||
config::Config,
|
||||
json_store::JSONStore,
|
||||
location::Location,
|
||||
routes::{api_error, FallibleApiResponse, Token},
|
||||
routes::{api_error, vec_to_api, FallibleApiResponse, ToAPI, Token},
|
||||
transaction::Transaction,
|
||||
};
|
||||
|
||||
|
@ -121,11 +120,6 @@ pub async fn location_inventory(
|
|||
}
|
||||
|
||||
Ok(json!(
|
||||
vec_to_api(
|
||||
&Transaction::in_location(location)
|
||||
.await
|
||||
.ok_or_else(|| api_error("No such location"))?
|
||||
)
|
||||
.await
|
||||
vec_to_api(&Transaction::in_location(location).await).await
|
||||
))
|
||||
}
|
||||
|
|
|
@ -4,12 +4,11 @@ mod location;
|
|||
mod stat;
|
||||
mod supply;
|
||||
|
||||
use std::str::FromStr;
|
||||
|
||||
pub use demand::*;
|
||||
pub use error::*;
|
||||
pub use location::*;
|
||||
use mongod::reference_of;
|
||||
use mongod::Model;
|
||||
use mongod::ToAPI;
|
||||
use rocket::post;
|
||||
use rocket::serde::json::Json;
|
||||
use serde::Deserialize;
|
||||
|
@ -25,7 +24,9 @@ use crate::check_auth;
|
|||
use crate::config::Config;
|
||||
use crate::db::get_items_without_min_satisfied;
|
||||
use crate::db::ItemDB;
|
||||
use crate::location::Location;
|
||||
use crate::get_locations;
|
||||
use crate::get_pg;
|
||||
use crate::routes::ToAPI;
|
||||
use crate::routes::Token;
|
||||
use crate::transaction::Transaction;
|
||||
|
||||
|
@ -91,7 +92,9 @@ pub async fn transaction_route(
|
|||
) -> FallibleApiResponse {
|
||||
check_auth!(t, c);
|
||||
|
||||
let t = Transaction::get(transaction)
|
||||
let t = Transaction::get(
|
||||
&uuid::Uuid::from_str(&transaction).map_err(|_| api_error("Invalid UUID"))?,
|
||||
)
|
||||
.await
|
||||
.ok_or_else(|| api_error("No transaction with this UUID"))?;
|
||||
Ok(t.api().await)
|
||||
|
@ -184,16 +187,16 @@ pub struct MoveTransaction {
|
|||
#[post("/transaction/<id>/move", data = "<form>")]
|
||||
pub async fn move_transaction_route(id: &str, form: Json<MoveTransaction>) -> FallibleApiResponse {
|
||||
let new_loc = &form.to;
|
||||
Transaction::get(id)
|
||||
.await
|
||||
.ok_or_else(|| api_error("No such transaction"))?
|
||||
.change()
|
||||
.location(if form.to.is_empty() {
|
||||
None
|
||||
} else {
|
||||
Some(reference_of!(Location, new_loc).ok_or_else(|| api_error("No such location"))?)
|
||||
})
|
||||
.update()
|
||||
let locations = get_locations!();
|
||||
|
||||
if !locations.contains_key(new_loc) {
|
||||
return Err(api_error("No such location"));
|
||||
}
|
||||
|
||||
sqlx::query("UPDATE transactions SET location = $1 WHERE id = $2")
|
||||
.bind(new_loc)
|
||||
.bind(uuid::Uuid::from_str(id).map_err(|_| api_error("Invalid UUID"))?)
|
||||
.execute(get_pg!())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
|
|
|
@ -73,7 +73,7 @@ pub async fn item_stat_route(
|
|||
let item_var = itemdb.get_item(&item).unwrap().variant(var).unwrap();
|
||||
for t in item_var.inventory().await {
|
||||
transaction_count += 1;
|
||||
total_price += t.price.value;
|
||||
total_price += t.price;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
use mongod::ToAPI;
|
||||
use rocket::serde::json::Json;
|
||||
use rocket::{get, post, State};
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
@ -6,7 +5,7 @@ use serde_json::json;
|
|||
|
||||
use crate::check_auth;
|
||||
use crate::config::{Config, Webhook};
|
||||
use crate::routes::Token;
|
||||
use crate::routes::{vec_to_api, ToAPI, Token};
|
||||
use crate::{
|
||||
db::ItemDB,
|
||||
routes::{api_error, FallibleApiResponse},
|
||||
|
@ -18,7 +17,7 @@ use super::{item_does_not_exist_error, variant_does_not_exist_error};
|
|||
pub struct SupplyForm {
|
||||
pub item: String,
|
||||
pub variant: String,
|
||||
pub price: String,
|
||||
pub price: f64,
|
||||
pub origin: Option<String>,
|
||||
pub location: Option<String>,
|
||||
pub note: Option<String>,
|
||||
|
@ -42,10 +41,7 @@ pub async fn supply_route(
|
|||
|
||||
let transaction = variant
|
||||
.supply(
|
||||
form.price
|
||||
.clone()
|
||||
.try_into()
|
||||
.map_err(|()| api_error("Price malformed"))?,
|
||||
form.price,
|
||||
form.origin.as_deref(),
|
||||
form.location.as_deref(),
|
||||
form.note.as_deref(),
|
||||
|
@ -58,7 +54,7 @@ pub async fn supply_route(
|
|||
}
|
||||
}
|
||||
|
||||
Ok(json!({"uuid": transaction._id}))
|
||||
Ok(json!({"uuid": transaction.id}))
|
||||
}
|
||||
|
||||
/// Returns a list of Transaction UUIDs for the Item Variant
|
||||
|
@ -104,7 +100,7 @@ pub async fn inventory_route(
|
|||
item.inventory().await
|
||||
};
|
||||
|
||||
Ok(json!(mongod::vec_to_api(&transactions).await))
|
||||
Ok(json!(vec_to_api(&transactions).await))
|
||||
}
|
||||
|
||||
/// Returns current active Transactions for Item Variant
|
||||
|
@ -126,7 +122,7 @@ pub async fn inventory_route_variant(
|
|||
|
||||
let transactions = variant.inventory().await;
|
||||
|
||||
Ok(json!(mongod::vec_to_api(&transactions).await))
|
||||
Ok(json!(vec_to_api(&transactions).await))
|
||||
}
|
||||
|
||||
/// Returns statistics for the Item Variant
|
||||
|
|
|
@ -37,3 +37,20 @@ impl<'r> FromRequest<'r> for Token {
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// A trait to generate a Model API representation in JSON format.
|
||||
pub trait ToAPI: Sized {
|
||||
/// Generate public API JSON
|
||||
fn api(&self) -> impl std::future::Future<Output = serde_json::Value>;
|
||||
}
|
||||
|
||||
/// Converts a slice of items implementing the `ToAPI` trait into a `Vec` of JSON values.
|
||||
pub async fn vec_to_api(items: &[impl ToAPI]) -> Vec<serde_json::Value> {
|
||||
let mut ret = Vec::with_capacity(items.len());
|
||||
|
||||
for e in items {
|
||||
ret.push(e.api().await);
|
||||
}
|
||||
|
||||
ret
|
||||
}
|
||||
|
|
|
@ -1,109 +1,85 @@
|
|||
use futures::StreamExt;
|
||||
use mongod::{
|
||||
assert_reference_of,
|
||||
derive::{Model, Referencable},
|
||||
reference_of, Model, Referencable, Reference, Sort, Validate,
|
||||
};
|
||||
use mongodb::bson::doc;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
use sqlx::prelude::FromRow;
|
||||
|
||||
use crate::{item::Item, location::Location};
|
||||
use crate::{get_itemdb, get_locations, get_pg, routes::ToAPI};
|
||||
|
||||
// todo : produced / consumed by flow field?
|
||||
|
||||
/// A Transaction of an Item Variant
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, Model, Referencable)]
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, FromRow)]
|
||||
pub struct Transaction {
|
||||
/// UUID
|
||||
pub _id: String,
|
||||
pub id: uuid::Uuid,
|
||||
/// Associated Item
|
||||
pub item: String,
|
||||
/// Associated Variant
|
||||
pub variant: String,
|
||||
/// Price of obtaining the Item
|
||||
pub price: Price,
|
||||
pub price: f64,
|
||||
/// Origin of the Item
|
||||
pub origin: Option<String>,
|
||||
/// The location of the Item
|
||||
pub location: Option<Reference>,
|
||||
/// Info on consumption of the Item
|
||||
pub consumed: Option<Consumed>,
|
||||
pub location: Option<String>,
|
||||
/// Notes on Transaction
|
||||
pub note: Option<String>,
|
||||
/// Timestamp of the Transaction
|
||||
pub timestamp: i64,
|
||||
}
|
||||
|
||||
impl Validate for Transaction {
|
||||
async fn validate(&self) -> Result<(), String> {
|
||||
if let Some(location) = &self.location {
|
||||
assert_reference_of!(location, Location);
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
/// Information about consumed Items
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Consumed {
|
||||
pub created: chrono::DateTime<chrono::Utc>,
|
||||
/// Destination of the Item or who consumed it
|
||||
pub destination: String,
|
||||
pub destination: Option<String>,
|
||||
/// Price the Item was exported or consumed at
|
||||
pub price: Price,
|
||||
pub consumed_price: Option<f64>,
|
||||
/// Timestamp of Consumption
|
||||
pub timestamp: i64,
|
||||
pub consumed_timestamp: Option<chrono::DateTime<chrono::Utc>>,
|
||||
}
|
||||
|
||||
impl Transaction {
|
||||
pub async fn new(
|
||||
item: &str,
|
||||
variant: &str,
|
||||
price: Price,
|
||||
price: f64,
|
||||
origin: Option<&str>,
|
||||
location: Option<&str>,
|
||||
note: Option<&str>,
|
||||
) -> Self {
|
||||
Self {
|
||||
_id: uuid::Uuid::new_v4().to_string(),
|
||||
item: item.to_string(),
|
||||
variant: variant.to_string(),
|
||||
price,
|
||||
consumed: None,
|
||||
origin: origin.map(std::string::ToString::to_string),
|
||||
location: if let Some(location) = location {
|
||||
reference_of!(Location, location)
|
||||
} else {
|
||||
None
|
||||
},
|
||||
note: note.map(|x| x.to_string()),
|
||||
timestamp: chrono::Utc::now().timestamp(),
|
||||
}
|
||||
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()
|
||||
}
|
||||
|
||||
/// Consumes the Item with `price` and `destination`
|
||||
pub async fn consume(self, price: Price, destination: &str) -> Self {
|
||||
self.change()
|
||||
.consumed(Some(Consumed {
|
||||
destination: destination.to_string(),
|
||||
price,
|
||||
timestamp: chrono::Utc::now().timestamp(),
|
||||
}))
|
||||
.update()
|
||||
pub async fn get(id: &uuid::Uuid) -> Option<Self> {
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE id = $1")
|
||||
.bind(id)
|
||||
.fetch_optional(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
/// 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()
|
||||
}
|
||||
|
||||
pub async fn is_expired_at(&self, time: i64) -> bool {
|
||||
if let Some(expiry) = Item::get(&self.item)
|
||||
.await
|
||||
if let Some(expiry) = get_itemdb!()
|
||||
.get_item(&self.item)
|
||||
.unwrap()
|
||||
.variant(&self.variant)
|
||||
.unwrap()
|
||||
.expiry
|
||||
{
|
||||
let date_added = self.timestamp;
|
||||
let date_added = self.created.timestamp();
|
||||
|
||||
let expiration_ts = expiry * 24 * 60 * 60;
|
||||
|
||||
|
@ -120,15 +96,16 @@ impl Transaction {
|
|||
}
|
||||
|
||||
pub async fn is_expired(&self) -> bool {
|
||||
if self.consumed.is_some() {
|
||||
if let Some(expiry) = Item::get(&self.item)
|
||||
.await
|
||||
if self.consumed_timestamp.is_some() {
|
||||
if let Some(expiry) = get_itemdb!()
|
||||
.get_item(&self.item)
|
||||
.unwrap()
|
||||
.variant(&self.variant)
|
||||
.unwrap()
|
||||
.expiry
|
||||
{
|
||||
let time_around = self.timestamp - self.consumed.as_ref().unwrap().timestamp;
|
||||
let time_around =
|
||||
self.created.timestamp() - self.consumed_timestamp.unwrap().timestamp();
|
||||
let expiration_ts = expiry * 24 * 60 * 60;
|
||||
return time_around > expiration_ts;
|
||||
} else {
|
||||
|
@ -140,38 +117,24 @@ impl Transaction {
|
|||
self.is_expired_at(current_time).await
|
||||
}
|
||||
|
||||
pub async fn in_location(l: &str) -> Option<Vec<Self>> {
|
||||
let l = reference_of!(Location, l)?;
|
||||
Some(
|
||||
Self::find(
|
||||
doc! { "location": l, "consumed": { "$not": { "$type": "object" } }},
|
||||
None,
|
||||
None,
|
||||
pub async fn in_location(l: &str) -> Vec<Self> {
|
||||
sqlx::query_as(
|
||||
"SELECT * FROM transactions WHERE location = $1 AND consumed_timestamp IS NULL",
|
||||
)
|
||||
.bind(l)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap(),
|
||||
)
|
||||
.unwrap()
|
||||
}
|
||||
|
||||
pub async fn in_location_recursive(l: &str) -> Option<Vec<Self>> {
|
||||
// get the children of this location
|
||||
let locations = Location::get(l).await?.children_recursive().await;
|
||||
let locations = get_locations!().get(l)?.children_recursive();
|
||||
|
||||
let l = reference_of!(Location, l)?;
|
||||
let mut transactions = Self::find(
|
||||
doc! { "location": l, "consumed": { "$not": { "$type": "object" } },},
|
||||
None,
|
||||
None,
|
||||
)
|
||||
.await
|
||||
.unwrap();
|
||||
let mut transactions = Self::in_location(l).await;
|
||||
|
||||
for loc in locations {
|
||||
transactions.extend(
|
||||
Self::find(doc! { "location": loc.reference(), "consumed": { "$not": { "$type": "object" } }}, None, None)
|
||||
.await
|
||||
.unwrap(),
|
||||
);
|
||||
transactions.extend(Self::in_location(&loc.id).await);
|
||||
}
|
||||
|
||||
Some(transactions)
|
||||
|
@ -179,13 +142,10 @@ impl Transaction {
|
|||
|
||||
/// Get all Transactions which are not consumed and are expired
|
||||
pub async fn active_expired(days: Option<i64>) -> Vec<Self> {
|
||||
let items = Self::find(
|
||||
doc! {
|
||||
"consumed": { "$not": { "$type": "object" } }
|
||||
},
|
||||
None,
|
||||
Some(doc! { "timestamp": Sort::Descending }),
|
||||
let items: Vec<Self> = sqlx::query_as(
|
||||
"SELECT * FROM transactions WHERE consumed_timestamp IS NULL ORDER BY created DESC",
|
||||
)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
|
@ -210,67 +170,35 @@ impl Transaction {
|
|||
}
|
||||
}
|
||||
|
||||
impl mongod::ToAPI for Transaction {
|
||||
impl ToAPI for Transaction {
|
||||
async fn api(&self) -> serde_json::Value {
|
||||
let location = if let Some(loc) = &self.location {
|
||||
Some(loc.get::<Location>().await.api().await)
|
||||
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()
|
||||
}))
|
||||
} else {
|
||||
None
|
||||
};
|
||||
|
||||
json!({
|
||||
"uuid": self._id,
|
||||
"uuid": self.id,
|
||||
"item": self.item,
|
||||
"variant": self.variant,
|
||||
"price": self.price,
|
||||
"origin": self.origin,
|
||||
"location": location,
|
||||
"timestamp": self.timestamp,
|
||||
"consumed": self.consumed,
|
||||
"timestamp": self.created.timestamp(),
|
||||
"consumed": consumed,
|
||||
"note": self.note,
|
||||
"expired": self.is_expired().await
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
/// Economic Price
|
||||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq)]
|
||||
pub struct Price {
|
||||
/// Value of the currency
|
||||
pub value: f64,
|
||||
/// Kind of currency
|
||||
pub currency: String,
|
||||
}
|
||||
|
||||
impl Price {
|
||||
pub fn new(value: f64, currency: &str) -> Self {
|
||||
Self {
|
||||
value,
|
||||
currency: currency.to_string(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn zero() -> Self {
|
||||
Self {
|
||||
value: 0.00,
|
||||
currency: String::new(),
|
||||
}
|
||||
}
|
||||
|
||||
fn parse(price: &str) -> Option<Self> {
|
||||
let (value, currency) = price.split_once(' ')?;
|
||||
|
||||
Some(Self {
|
||||
value: value.parse().ok()?,
|
||||
currency: currency.to_string(),
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl TryFrom<String> for Price {
|
||||
type Error = ();
|
||||
|
||||
fn try_from(value: String) -> Result<Self, Self::Error> {
|
||||
Self::parse(&value).ok_or(())
|
||||
}
|
||||
}
|
||||
|
|
245
src/variant.rs
245
src/variant.rs
|
@ -1,16 +1,9 @@
|
|||
use std::collections::HashMap;
|
||||
|
||||
use futures::StreamExt;
|
||||
use mongod::{Model, Sort};
|
||||
use mongodb::bson::doc;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use serde_json::json;
|
||||
|
||||
use crate::transaction::{Price, Transaction};
|
||||
|
||||
pub fn sort_by_timestamp() -> mongodb::bson::Document {
|
||||
doc! { "timestamp": mongod::Sort::Descending }
|
||||
}
|
||||
use crate::{get_pg, transaction::Transaction};
|
||||
|
||||
pub fn timestamp_range(year: i32, month: u32) -> (i64, i64) {
|
||||
let d = chrono::NaiveDate::from_ymd_opt(year, month, 0).unwrap();
|
||||
|
@ -85,69 +78,51 @@ impl Variant {
|
|||
|
||||
/// Returns the IDs of Transactions from this Item Variant.
|
||||
pub async fn supply_log(&self) -> Vec<String> {
|
||||
let filter = doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant
|
||||
};
|
||||
|
||||
let result = Transaction::find_partial(filter, json!({}), None, None)
|
||||
let res: Vec<(uuid::Uuid,)> = sqlx::query_as(
|
||||
"SELECT id FROM transactions WHERE item = $1 AND variant = $2 ORDER BY created DESC",
|
||||
)
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let mut ret = Vec::new();
|
||||
|
||||
for doc in result {
|
||||
ret.push(doc._id);
|
||||
}
|
||||
|
||||
ret
|
||||
res.into_iter().map(|x| x.0.to_string()).collect()
|
||||
}
|
||||
|
||||
/// Returns the active Transaction of this Item Variant which are not yet consumed.
|
||||
pub async fn inventory(&self) -> Vec<Transaction> {
|
||||
let filter = doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant,
|
||||
"consumed": { "$not": { "$type": "object" } }
|
||||
};
|
||||
|
||||
Transaction::find(filter, None, None).await.unwrap()
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE item = $1 AND variant = $2 AND consumed_timestamp IS NULL ORDER BY created DESC")
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
}
|
||||
|
||||
/// Returns the IDs of the Transactions from this Item Variant which are consumed.
|
||||
pub async fn demand_log(&self, destination: Option<&str>) -> Vec<String> {
|
||||
let filter = if let Some(dest) = destination {
|
||||
doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant,
|
||||
"consumed": { "destination": dest }
|
||||
}
|
||||
let res: Vec<(uuid::Uuid,)> = if let Some(destination) = destination {
|
||||
sqlx::query_as(
|
||||
"SELECT id FROM transactions WHERE item = $1 AND variant = $2 AND consumed_timestamp IS NOT NULL AND destination = $3 ORDER BY created DESC"
|
||||
)
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.bind(destination)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
} else {
|
||||
doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant,
|
||||
"consumed": { "$type": "object" }
|
||||
}
|
||||
sqlx::query_as(
|
||||
"SELECT id FROM transactions WHERE item = $1 AND variant = $2 AND consumed_timestamp IS NOT NULL ORDER BY created DESC"
|
||||
)
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
};
|
||||
|
||||
let result = Transaction::find_partial(filter, json!({}), None, None)
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let mut ret = Vec::new();
|
||||
|
||||
for doc in result {
|
||||
ret.push(doc._id);
|
||||
res.into_iter().map(|x| x.0.to_string()).collect()
|
||||
}
|
||||
|
||||
ret
|
||||
}
|
||||
|
||||
pub async fn demand(uuid: &str, price: Price, destination: &str) -> Option<Transaction> {
|
||||
pub async fn demand(uuid: &uuid::Uuid, price: f64, destination: &str) -> Option<Transaction> {
|
||||
// check if transaction exists
|
||||
let mut t = Transaction::get(uuid).await?;
|
||||
t = t.consume(price, destination).await;
|
||||
Some(t)
|
||||
let t = Transaction::get(uuid).await?;
|
||||
Some(t.consume(price, destination).await)
|
||||
}
|
||||
|
||||
/// Records a supply transaction in the database.
|
||||
|
@ -162,26 +137,22 @@ impl Variant {
|
|||
/// Returns a UUID string representing the transaction.
|
||||
pub async fn supply(
|
||||
&self,
|
||||
price: Price,
|
||||
price: f64,
|
||||
origin: Option<&str>,
|
||||
location: Option<&str>,
|
||||
note: Option<&str>,
|
||||
) -> Transaction {
|
||||
let t = Transaction::new(&self.item, &self.variant, price, origin, location, note).await;
|
||||
|
||||
t.insert().await.unwrap();
|
||||
|
||||
t
|
||||
Transaction::new(&self.item, &self.variant, price, origin, location, note).await
|
||||
}
|
||||
|
||||
/// Returns all Transactions of this Item Variant
|
||||
pub async fn get_all_transactions(&self) -> Vec<Transaction> {
|
||||
let filter = doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant
|
||||
};
|
||||
|
||||
Transaction::find(filter, None, Some(doc! { "timestamp": Sort::Descending }))
|
||||
sqlx::query_as(
|
||||
"SELECT * FROM transactions WHERE item = $1 AND variant = $2 ORDER BY created DESC",
|
||||
)
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap()
|
||||
}
|
||||
|
@ -189,80 +160,57 @@ impl Variant {
|
|||
pub async fn get_transaction_timeslice(&self, year: i32, month: u32) -> Vec<Transaction> {
|
||||
let (start, end) = timestamp_range(year, month);
|
||||
|
||||
Transaction::find(
|
||||
doc! {
|
||||
"timestamp": {
|
||||
"$gte": start,
|
||||
"$lte": end
|
||||
}
|
||||
},
|
||||
None,
|
||||
Some(sort_by_timestamp()),
|
||||
)
|
||||
.await
|
||||
.unwrap()
|
||||
sqlx::query_as("SELECT * FROM transactions WHERE created BETWEEN to_timestamp($1) AND to_timestamp($2) ORDER BY created DESC")
|
||||
.bind(start)
|
||||
.bind(end)
|
||||
.fetch_all(get_pg!()).await.unwrap()
|
||||
}
|
||||
|
||||
pub async fn get_unique_origins(&self) -> Vec<String> {
|
||||
unique_flows(
|
||||
&Transaction::unique(
|
||||
doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant
|
||||
},
|
||||
"origin",
|
||||
)
|
||||
.await,
|
||||
)
|
||||
let res: Vec<(String,)> = sqlx::query_as("SELECT DISTINCT(origin) FROM transactions WHERE origin NOT LIKE 'flow::%' AND item = $1 AND variant = $2")
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!()).await.unwrap();
|
||||
res.into_iter().map(|x| x.0).collect()
|
||||
}
|
||||
|
||||
pub async fn get_unique_destinations(&self) -> Vec<String> {
|
||||
unique_flows(
|
||||
&Transaction::unique(
|
||||
doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant
|
||||
},
|
||||
"consumed.destination",
|
||||
)
|
||||
.await,
|
||||
)
|
||||
let res: Vec<(String,)> = sqlx::query_as("SELECT DISTINCT(destination) FROM transactions WHERE destination NOT LIKE 'flow::%' AND item = $1 AND variant = $2")
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!()).await.unwrap();
|
||||
res.into_iter().map(|x| x.0).collect()
|
||||
}
|
||||
|
||||
pub async fn price_history_by_origin(&self, origin: &str, limit: Option<i64>) -> Vec<Price> {
|
||||
Transaction::find(
|
||||
doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant,
|
||||
"origin": origin
|
||||
},
|
||||
limit,
|
||||
Some(sort_by_timestamp()),
|
||||
pub async fn price_history_by_origin(&self, origin: &str, limit: Option<i64>) -> Vec<f64> {
|
||||
let res: Vec<(f64,)> = sqlx::query_as(
|
||||
&format!("SELECT price FROM transactions WHERE item = $1 AND variant = $2 AND origin = $3 ORDER BY created DESC {}", if let Some(limit) = limit {
|
||||
format!("LIMIT {limit}")
|
||||
} else { String::new() })
|
||||
)
|
||||
.await
|
||||
.unwrap()
|
||||
.into_iter()
|
||||
.map(|x| x.price)
|
||||
.collect()
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.bind(origin)
|
||||
.fetch_all(get_pg!()).await.unwrap();
|
||||
res.into_iter().map(|x| x.0).collect()
|
||||
}
|
||||
|
||||
pub async fn get_latest_price(&self, origin: Option<String>) -> Price {
|
||||
let mut filter = doc! {
|
||||
"item": &self.item,
|
||||
"variant": &self.variant
|
||||
};
|
||||
|
||||
pub async fn get_latest_price(&self, origin: Option<String>) -> f64 {
|
||||
if let Some(origin) = origin {
|
||||
filter.insert("origin", origin);
|
||||
let res: (f64,) = sqlx::query_as("SELECT price FROM transactions WHERE item = $1 AND variant = $2 AND origin = $3 ORDER BY created DESC LIMIT 1")
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.bind(origin)
|
||||
.fetch_one(get_pg!()).await.unwrap();
|
||||
res.0
|
||||
} else {
|
||||
let res: (f64,) = sqlx::query_as("SELECT price FROM transactions WHERE item = $1 AND variant = $2 ORDER BY created DESC LIMIT 1")
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.bind(origin)
|
||||
.fetch_one(get_pg!()).await.unwrap();
|
||||
res.0
|
||||
}
|
||||
|
||||
Transaction::find(filter, Some(1), Some(sort_by_timestamp()))
|
||||
.await
|
||||
.unwrap()
|
||||
.first()
|
||||
.unwrap()
|
||||
.price
|
||||
.clone()
|
||||
}
|
||||
|
||||
/// Check if item variant is below minimum. Returns if this is the case and the number needed to fulfill minimum
|
||||
|
@ -280,8 +228,7 @@ impl Variant {
|
|||
pub async fn stat(&self, full: bool) -> serde_json::Value {
|
||||
let active_transactions = self.inventory().await;
|
||||
|
||||
// fix : ignores currency
|
||||
let total_price: f64 = active_transactions.iter().map(|x| x.price.value).sum();
|
||||
let total_price: f64 = active_transactions.iter().map(|x| x.price).sum();
|
||||
|
||||
if !full {
|
||||
return json!({
|
||||
|
@ -290,13 +237,15 @@ impl Variant {
|
|||
});
|
||||
}
|
||||
|
||||
let all_transactions = Transaction::find(
|
||||
doc! { "item": &self.item, "variant": &self.variant},
|
||||
None,
|
||||
None,
|
||||
let all_transactions: Vec<Transaction> = sqlx::query_as(
|
||||
"SELECT * FROM transactions WHERE item = $1 AND variant = $2 ORDER BY created DESC",
|
||||
)
|
||||
.bind(&self.item)
|
||||
.bind(&self.variant)
|
||||
.fetch_all(get_pg!())
|
||||
.await
|
||||
.unwrap();
|
||||
|
||||
let mut expired_count = 0.0;
|
||||
|
||||
for t in &all_transactions {
|
||||
|
@ -318,7 +267,7 @@ impl Variant {
|
|||
.price_history_by_origin(&origin, None)
|
||||
.await
|
||||
.into_iter()
|
||||
.map(|x| x.value)
|
||||
.map(|x| x)
|
||||
.collect::<Vec<_>>();
|
||||
let prices_len = prices.len() as f64;
|
||||
let prices_summed = prices.into_iter().reduce(|acc, e| acc + e).unwrap_or(0.0);
|
||||
|
@ -349,29 +298,3 @@ impl Variant {
|
|||
})
|
||||
}
|
||||
}
|
||||
|
||||
pub fn unique_flows(i: &[String]) -> Vec<String> {
|
||||
let mut unique_vec: Vec<String> = Vec::new();
|
||||
|
||||
for s in i {
|
||||
// Check if the string starts with "flow::"
|
||||
if let Some(suffix) = s.strip_prefix("flow::") {
|
||||
// Extract the part after "flow::" and split on "::" to get the kind (ignoring id)
|
||||
let parts: Vec<&str> = suffix.split("::").collect();
|
||||
if let Some(kind) = parts.first() {
|
||||
// Build the common prefix "flow::kind"
|
||||
let common_prefix = format!("flow::{}", kind);
|
||||
|
||||
if !unique_vec.contains(&common_prefix) {
|
||||
unique_vec.push(common_prefix);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// If the string doesn't start with "flow::", retain it
|
||||
// Assumption: Except "flow::" values, everything should be already unique
|
||||
unique_vec.push(s.to_string());
|
||||
}
|
||||
}
|
||||
|
||||
unique_vec
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue