use crate::{ db::DBPool, models::{item::Item, task::Task}, services::{spawn, urges}, DResult, }; use log::info; use std::collections::{BTreeMap, BTreeSet}; pub mod dynzone; pub mod fixed_item; pub mod journals; pub mod npc; pub mod possession_type; pub mod room; pub mod species; pub struct StaticItem { pub item_code: &'static str, pub initial_item: Box Item>, pub extra_items_on_create: Box Box>>, } pub struct StaticTask { pub task_code: String, pub initial_task: Box Task>, } struct StaticThingTypeGroup { thing_type: &'static str, things: fn() -> Box>, } fn static_item_registry() -> Vec> { vec![ // Must have no duplicates. StaticThingTypeGroup:: { thing_type: "npc", things: || npc::npc_static_items(), }, StaticThingTypeGroup:: { thing_type: "room", things: || room::room_static_items(), }, StaticThingTypeGroup:: { thing_type: "fixed_item", things: || fixed_item::static_items(), }, ] } fn static_task_registry() -> Vec> { vec![ // Must have no duplicates. StaticThingTypeGroup:: { thing_type: "NPCSay", things: || npc::npc_say_tasks(), }, StaticThingTypeGroup:: { thing_type: "NPCWander", things: || npc::npc_wander_tasks(), }, StaticThingTypeGroup:: { thing_type: "NPCAggro", things: || npc::npc_aggro_tasks(), }, StaticThingTypeGroup:: { thing_type: "TickUrges", things: || urges::urge_tasks(), }, StaticThingTypeGroup:: { thing_type: "ResetSpawns", things: || spawn::reset_tasks(), }, ] } async fn refresh_static_items(pool: &DBPool) -> DResult<()> { let registry = static_item_registry(); let expected_type: BTreeSet = registry.iter().map(|x| x.thing_type.to_owned()).collect(); let cur_types: Box> = pool.find_static_item_types().await?; for item_type in cur_types.difference(&expected_type) { pool.delete_static_items_by_type(item_type).await?; } for type_group in registry.iter() { info!( "Checking static_content of item_type {}", type_group.thing_type ); let tx = pool.start_transaction().await?; let existing_items = tx.find_static_items_by_type(type_group.thing_type).await?; let expected_items: BTreeMap = (type_group.things)() .map(|x| (x.item_code.to_owned(), x)) .collect(); let expected_set: BTreeSet = expected_items.keys().map(|x| x.to_owned()).collect(); for unwanted_item in existing_items.difference(&expected_set) { info!("Deleting item {:?}", unwanted_item); tx.delete_static_items_by_code(type_group.thing_type, unwanted_item) .await?; } for new_item_code in expected_set.difference(&existing_items) { info!("Creating item {:?}", new_item_code); let expected_item = expected_items.get(new_item_code).unwrap(); let new_item = &(expected_item.initial_item)(); tx.create_item(new_item).await?; for mut item in (expected_item.extra_items_on_create)(&new_item) { item.item_code = format!("{}", tx.alloc_item_code().await?); tx.create_item(&item).await?; } } for existing_item_code in expected_set.intersection(&existing_items) { tx.limited_update_static_item(&(expected_items .get(existing_item_code) .unwrap() .initial_item)()) .await?; } tx.commit().await?; info!( "Committed any changes for static_content of item_type {}", type_group.thing_type ); } Ok(()) } async fn refresh_static_tasks(pool: &DBPool) -> DResult<()> { let registry = static_task_registry(); let expected_type: BTreeSet = registry.iter().map(|x| x.thing_type.to_owned()).collect(); let cur_types: Box> = pool.find_static_task_types().await?; for task_type in cur_types.difference(&expected_type) { pool.delete_static_tasks_by_type(task_type).await?; } for type_group in registry.iter() { info!( "Checking static_content of task_type {}", type_group.thing_type ); let tx = pool.start_transaction().await?; let existing_tasks = tx.find_static_tasks_by_type(type_group.thing_type).await?; let expected_tasks: BTreeMap = (type_group.things)() .map(|x| (x.task_code.to_owned(), x)) .collect(); let expected_set: BTreeSet = expected_tasks.keys().map(|x| x.to_owned()).collect(); for unwanted_task in existing_tasks.difference(&expected_set) { info!("Deleting task {:?}", unwanted_task); tx.delete_static_tasks_by_code(type_group.thing_type, unwanted_task) .await?; } for new_task_code in expected_set.difference(&existing_tasks) { info!("Creating task {:?}", new_task_code); tx.upsert_task(&(expected_tasks.get(new_task_code).unwrap().initial_task)()) .await?; } for existing_task_code in expected_set.intersection(&existing_tasks) { tx.limited_update_static_task(&(expected_tasks .get(existing_task_code) .unwrap() .initial_task)()) .await?; } tx.commit().await?; info!( "Committed any changes for static_content of task_type {}", type_group.thing_type ); } Ok(()) } pub async fn refresh_static_content(pool: &DBPool) -> DResult<()> { refresh_static_items(pool).await?; refresh_static_tasks(pool).await?; Ok(()) } #[cfg(test)] mod test { use super::*; use itertools::Itertools; #[test] fn no_duplicate_static_items() { let mut registry = static_item_registry(); registry.sort_unstable_by(|x, y| x.thing_type.cmp(y.thing_type)); let duplicates: Vec<&'static str> = registry .iter() .group_by(|x| x.thing_type) .into_iter() .filter_map(|(k, v)| if v.count() <= 1 { None } else { Some(k) }) .collect(); if duplicates.len() > 0 { panic!( "static_item_registry has duplicate item_types: {:}", duplicates.join(", ") ); } for type_group in registry.iter() { let iterator: Box> = (type_group.things)(); let duplicates: Vec<&'static str> = iterator .group_by(|x| x.item_code) .into_iter() .filter_map(|(k, v)| if v.count() <= 1 { None } else { Some(k) }) .collect(); if duplicates.len() > 0 { panic!( "static_item_registry has duplicate item_codes for {}: {:}", type_group.thing_type, duplicates.join(", ") ); } } } #[test] fn no_duplicate_static_tasks() { let mut registry = static_task_registry(); registry.sort_unstable_by(|x, y| x.thing_type.cmp(y.thing_type)); let duplicates: Vec<&'static str> = registry .iter() .group_by(|x| x.thing_type) .into_iter() .filter_map(|(k, v)| if v.count() <= 1 { None } else { Some(k) }) .collect(); if duplicates.len() > 0 { panic!( "static_task_registry has duplicate task_types: {:}", duplicates.join(", ") ); } for type_group in registry.iter() { let iterator: Box> = (type_group.things)(); let duplicates: Vec = iterator .group_by(|x| x.task_code.clone()) .into_iter() .filter_map(|(k, v)| if v.count() <= 1 { None } else { Some(k) }) .collect(); if duplicates.len() > 0 { panic!( "static_task_registry has duplicate task_codes for {}: {:}", type_group.thing_type, duplicates.join(", ") ); } } } }