2022-03-02 21:03:28 -05:00
|
|
|
// Copyright 2022 Jeremy Wall
|
2022-03-06 18:04:49 -05:00
|
|
|
//
|
2022-03-02 21:03:28 -05:00
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
2022-03-06 18:04:49 -05:00
|
|
|
//
|
2022-03-02 21:03:28 -05:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2022-03-06 18:04:49 -05:00
|
|
|
//
|
2022-03-02 21:03:28 -05:00
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
2022-10-21 14:57:50 -04:00
|
|
|
use std::collections::{BTreeMap, BTreeSet};
|
|
|
|
|
2022-12-22 10:49:50 -05:00
|
|
|
use client_api::UserData;
|
2022-12-28 12:59:11 -06:00
|
|
|
use recipes::{parse, Ingredient, IngredientAccumulator, IngredientKey, Recipe, RecipeEntry};
|
|
|
|
use serde_json::from_str;
|
|
|
|
use sycamore::futures::spawn_local_scoped;
|
2022-12-28 14:27:45 -06:00
|
|
|
use sycamore::prelude::*;
|
2022-12-23 13:34:40 -05:00
|
|
|
use sycamore_state::{Handler, MessageMapper};
|
2022-12-28 12:59:11 -06:00
|
|
|
use tracing::{debug, error, info, instrument, warn};
|
2022-12-23 13:34:40 -05:00
|
|
|
|
2022-12-28 11:54:24 -06:00
|
|
|
use crate::api::HttpStore;
|
2022-12-28 12:59:11 -06:00
|
|
|
use crate::js_lib;
|
2022-12-28 11:54:24 -06:00
|
|
|
|
2022-12-23 13:34:40 -05:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub struct AppState {
|
|
|
|
pub recipe_counts: BTreeMap<String, usize>,
|
|
|
|
pub extras: BTreeSet<(String, String)>,
|
|
|
|
pub staples: Option<Recipe>,
|
|
|
|
pub recipes: BTreeMap<String, Recipe>,
|
|
|
|
pub category_map: BTreeMap<String, String>,
|
|
|
|
pub filtered_ingredients: BTreeSet<IngredientKey>,
|
|
|
|
pub modified_amts: BTreeMap<IngredientKey, String>,
|
|
|
|
pub auth: Option<UserData>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl AppState {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
recipe_counts: BTreeMap::new(),
|
|
|
|
extras: BTreeSet::new(),
|
|
|
|
staples: None,
|
|
|
|
recipes: BTreeMap::new(),
|
|
|
|
category_map: BTreeMap::new(),
|
|
|
|
filtered_ingredients: BTreeSet::new(),
|
|
|
|
modified_amts: BTreeMap::new(),
|
|
|
|
auth: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-28 11:54:24 -06:00
|
|
|
#[derive(Debug)]
|
2022-12-23 13:34:40 -05:00
|
|
|
pub enum Message {
|
|
|
|
InitRecipeCounts(BTreeMap<String, usize>),
|
|
|
|
UpdateRecipeCount(String, usize),
|
|
|
|
InitExtras(BTreeSet<(String, String)>),
|
|
|
|
AddExtra(String, String),
|
|
|
|
RemoveExtra(String, String),
|
|
|
|
InitRecipes(BTreeMap<String, Recipe>),
|
2022-12-28 14:27:45 -06:00
|
|
|
SaveRecipe(RecipeEntry),
|
2022-12-23 13:34:40 -05:00
|
|
|
SetRecipe(String, Recipe),
|
|
|
|
RemoveRecipe(String),
|
|
|
|
SetStaples(Option<Recipe>),
|
|
|
|
SetCategoryMap(BTreeMap<String, String>),
|
|
|
|
InitFilteredIngredient(BTreeSet<IngredientKey>),
|
|
|
|
AddFilteredIngredient(IngredientKey),
|
|
|
|
RemoveFilteredIngredient(IngredientKey),
|
|
|
|
InitAmts(BTreeMap<IngredientKey, String>),
|
|
|
|
UpdateAmt(IngredientKey, String),
|
|
|
|
SetUserData(UserData),
|
|
|
|
UnsetUserData,
|
2022-12-28 11:54:24 -06:00
|
|
|
SaveState,
|
2022-12-28 12:59:11 -06:00
|
|
|
LoadState,
|
2022-12-23 13:34:40 -05:00
|
|
|
}
|
|
|
|
|
2022-12-28 11:54:24 -06:00
|
|
|
pub struct StateMachine(HttpStore);
|
2022-12-23 13:34:40 -05:00
|
|
|
|
2022-12-28 12:59:11 -06:00
|
|
|
#[instrument]
|
|
|
|
fn filter_recipes(
|
|
|
|
recipe_entries: &Option<Vec<RecipeEntry>>,
|
|
|
|
) -> Result<(Option<Recipe>, Option<BTreeMap<String, Recipe>>), String> {
|
|
|
|
match recipe_entries {
|
|
|
|
Some(parsed) => {
|
|
|
|
let mut staples = None;
|
|
|
|
let mut parsed_map = BTreeMap::new();
|
|
|
|
for r in parsed {
|
|
|
|
let recipe = match parse::as_recipe(&r.recipe_text()) {
|
|
|
|
Ok(r) => r,
|
|
|
|
Err(e) => {
|
|
|
|
error!("Error parsing recipe {}", e);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if recipe.title == "Staples" {
|
|
|
|
staples = Some(recipe);
|
|
|
|
} else {
|
|
|
|
parsed_map.insert(r.recipe_id().to_owned(), recipe);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok((staples, Some(parsed_map)))
|
|
|
|
}
|
|
|
|
None => Ok((None, None)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl StateMachine {
|
|
|
|
async fn load_state(store: HttpStore, original: &Signal<AppState>) {
|
|
|
|
let mut state = original.get().as_ref().clone();
|
|
|
|
info!("Synchronizing Recipes");
|
|
|
|
// TODO(jwall): Make our caching logic using storage more robust.
|
|
|
|
let recipe_entries = match store.get_recipes().await {
|
|
|
|
Ok(recipe_entries) => {
|
|
|
|
if let Ok((staples, recipes)) = filter_recipes(&recipe_entries) {
|
|
|
|
state.staples = staples;
|
|
|
|
if let Some(recipes) = recipes {
|
|
|
|
state.recipes = recipes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
recipe_entries
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
error!(?err);
|
|
|
|
None
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Ok(Some(plan)) = store.get_plan().await {
|
|
|
|
// set the counts.
|
|
|
|
let mut plan_map = BTreeMap::new();
|
|
|
|
for (id, count) in plan {
|
|
|
|
plan_map.insert(id, count as usize);
|
|
|
|
}
|
|
|
|
state.recipe_counts = plan_map;
|
|
|
|
} else {
|
|
|
|
// Initialize things to zero
|
|
|
|
if let Some(rs) = recipe_entries {
|
|
|
|
for r in rs {
|
|
|
|
state.recipe_counts.insert(r.recipe_id().to_owned(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info!("Checking for user_data in local storage");
|
|
|
|
let storage = js_lib::get_storage();
|
|
|
|
let user_data = storage
|
|
|
|
.get("user_data")
|
|
|
|
.expect("Couldn't read from storage");
|
|
|
|
if let Some(data) = user_data {
|
|
|
|
if let Ok(user_data) = from_str(&data) {
|
|
|
|
state.auth = Some(user_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info!("Synchronizing categories");
|
|
|
|
match store.get_categories().await {
|
|
|
|
Ok(Some(categories_content)) => {
|
|
|
|
debug!(categories=?categories_content);
|
|
|
|
match recipes::parse::as_categories(&categories_content) {
|
|
|
|
Ok(category_map) => {
|
|
|
|
state.category_map = category_map;
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
error!(?err)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
Ok(None) => {
|
|
|
|
warn!("There is no category file");
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
error!("{:?}", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info!("Synchronizing inventory data");
|
|
|
|
match store.get_inventory_data().await {
|
|
|
|
Ok((filtered_ingredients, modified_amts, extra_items)) => {
|
|
|
|
state.modified_amts = modified_amts;
|
|
|
|
state.filtered_ingredients = filtered_ingredients;
|
|
|
|
state.extras = BTreeSet::from_iter(extra_items);
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
error!("{:?}", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
original.set(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-23 13:34:40 -05:00
|
|
|
impl MessageMapper<Message, AppState> for StateMachine {
|
2022-12-28 11:54:24 -06:00
|
|
|
#[instrument(skip_all, fields(?msg))]
|
2022-12-28 12:59:11 -06:00
|
|
|
fn map<'ctx>(&self, cx: Scope<'ctx>, msg: Message, original: &'ctx Signal<AppState>) {
|
2022-12-23 13:34:40 -05:00
|
|
|
let mut original_copy = original.get().as_ref().clone();
|
|
|
|
match msg {
|
|
|
|
Message::InitRecipeCounts(map) => {
|
|
|
|
original_copy.recipe_counts = map;
|
|
|
|
}
|
|
|
|
Message::UpdateRecipeCount(id, count) => {
|
|
|
|
original_copy.recipe_counts.insert(id, count);
|
|
|
|
}
|
|
|
|
Message::InitExtras(set) => {
|
|
|
|
original_copy.extras = set;
|
|
|
|
}
|
|
|
|
Message::AddExtra(amt, name) => {
|
|
|
|
original_copy.extras.insert((amt, name));
|
|
|
|
}
|
|
|
|
Message::RemoveExtra(amt, name) => {
|
|
|
|
original_copy.extras.remove(&(amt, name));
|
|
|
|
}
|
|
|
|
Message::SetStaples(staples) => {
|
|
|
|
original_copy.staples = staples;
|
|
|
|
}
|
|
|
|
Message::InitRecipes(recipes) => {
|
|
|
|
original_copy.recipes = recipes;
|
|
|
|
}
|
|
|
|
Message::SetRecipe(id, recipe) => {
|
|
|
|
original_copy.recipes.insert(id, recipe);
|
|
|
|
}
|
2022-12-28 14:27:45 -06:00
|
|
|
Message::SaveRecipe(entry) => {
|
|
|
|
let recipe =
|
|
|
|
parse::as_recipe(entry.recipe_text()).expect("Failed to parse RecipeEntry");
|
|
|
|
original_copy
|
|
|
|
.recipes
|
|
|
|
.insert(entry.recipe_id().to_owned(), recipe);
|
|
|
|
let store = self.0.clone();
|
|
|
|
original_copy
|
|
|
|
.recipe_counts
|
|
|
|
.insert(entry.recipe_id().to_owned(), 0);
|
|
|
|
spawn_local_scoped(cx, async move {
|
|
|
|
if let Err(e) = store.save_recipes(vec![entry]).await {
|
|
|
|
error!(err=?e, "Unable to save Recipe");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2022-12-23 13:34:40 -05:00
|
|
|
Message::RemoveRecipe(id) => {
|
|
|
|
original_copy.recipes.remove(&id);
|
|
|
|
}
|
|
|
|
Message::SetCategoryMap(map) => {
|
|
|
|
original_copy.category_map = map;
|
|
|
|
}
|
|
|
|
Message::InitFilteredIngredient(set) => {
|
|
|
|
original_copy.filtered_ingredients = set;
|
|
|
|
}
|
|
|
|
Message::AddFilteredIngredient(key) => {
|
|
|
|
original_copy.filtered_ingredients.insert(key);
|
|
|
|
}
|
|
|
|
Message::RemoveFilteredIngredient(key) => {
|
|
|
|
original_copy.filtered_ingredients.remove(&key);
|
|
|
|
}
|
|
|
|
Message::InitAmts(map) => {
|
|
|
|
original_copy.modified_amts = map;
|
|
|
|
}
|
|
|
|
Message::UpdateAmt(key, amt) => {
|
|
|
|
original_copy.modified_amts.insert(key, amt);
|
|
|
|
}
|
|
|
|
Message::SetUserData(user_data) => {
|
|
|
|
original_copy.auth = Some(user_data);
|
|
|
|
}
|
|
|
|
Message::UnsetUserData => {
|
|
|
|
original_copy.auth = None;
|
|
|
|
}
|
2022-12-28 11:54:24 -06:00
|
|
|
Message::SaveState => {
|
|
|
|
let store = self.0.clone();
|
|
|
|
let original_copy = original_copy.clone();
|
2022-12-28 12:59:11 -06:00
|
|
|
spawn_local_scoped(cx, async move {
|
2022-12-28 11:54:24 -06:00
|
|
|
if let Err(e) = store.save_app_state(original_copy).await {
|
|
|
|
error!(err=?e, "Error saving app state")
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
2022-12-28 12:59:11 -06:00
|
|
|
Message::LoadState => {
|
|
|
|
let store = self.0.clone();
|
|
|
|
spawn_local_scoped(cx, async move {
|
|
|
|
Self::load_state(store, original).await;
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
2022-12-23 13:34:40 -05:00
|
|
|
}
|
2022-12-28 12:59:11 -06:00
|
|
|
original.set(original_copy);
|
2022-12-23 13:34:40 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-26 21:29:09 -05:00
|
|
|
pub type StateHandler<'ctx> = &'ctx Handler<'ctx, StateMachine, AppState, Message>;
|
|
|
|
|
2022-12-28 11:54:24 -06:00
|
|
|
pub fn get_state_handler<'ctx>(
|
|
|
|
cx: Scope<'ctx>,
|
|
|
|
initial: AppState,
|
|
|
|
store: HttpStore,
|
|
|
|
) -> StateHandler<'ctx> {
|
|
|
|
Handler::new(cx, initial, StateMachine(store))
|
2022-12-23 13:34:40 -05:00
|
|
|
}
|
2022-12-28 11:54:24 -06:00
|
|
|
|
2022-11-24 10:28:02 -05:00
|
|
|
#[derive(Debug)]
|
2022-10-21 14:57:50 -04:00
|
|
|
pub struct State {
|
2022-11-11 16:45:14 -05:00
|
|
|
pub recipe_counts: RcSignal<BTreeMap<String, RcSignal<usize>>>,
|
2022-10-25 15:44:56 -04:00
|
|
|
pub extras: RcSignal<Vec<(usize, (RcSignal<String>, RcSignal<String>))>>,
|
2022-10-21 14:57:50 -04:00
|
|
|
pub staples: RcSignal<Option<Recipe>>,
|
|
|
|
pub recipes: RcSignal<BTreeMap<String, Recipe>>,
|
|
|
|
pub category_map: RcSignal<BTreeMap<String, String>>,
|
2022-11-11 17:35:10 -05:00
|
|
|
pub filtered_ingredients: RcSignal<BTreeSet<IngredientKey>>,
|
|
|
|
pub modified_amts: RcSignal<BTreeMap<IngredientKey, RcSignal<String>>>,
|
2022-12-22 10:49:50 -05:00
|
|
|
pub auth: RcSignal<Option<UserData>>,
|
2022-10-21 14:57:50 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl State {
|
|
|
|
pub fn get_from_context(cx: Scope) -> std::rc::Rc<Self> {
|
|
|
|
use_context::<std::rc::Rc<Self>>(cx).clone()
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:45:14 -05:00
|
|
|
pub fn get_menu_list(&self) -> Vec<(String, RcSignal<usize>)> {
|
2022-10-21 14:57:50 -04:00
|
|
|
self.recipe_counts
|
|
|
|
.get()
|
|
|
|
.iter()
|
2022-11-11 16:45:14 -05:00
|
|
|
.map(|(k, v)| (k.clone(), v.clone()))
|
|
|
|
.filter(|(_, v)| *(v.get_untracked()) != 0)
|
2022-10-21 14:57:50 -04:00
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[instrument(skip(self))]
|
|
|
|
pub fn get_shopping_list(
|
|
|
|
&self,
|
|
|
|
show_staples: bool,
|
|
|
|
) -> BTreeMap<String, Vec<(Ingredient, BTreeSet<String>)>> {
|
|
|
|
let mut acc = IngredientAccumulator::new();
|
|
|
|
let recipe_counts = self.get_menu_list();
|
|
|
|
for (idx, count) in recipe_counts.iter() {
|
2022-11-11 16:45:14 -05:00
|
|
|
for _ in 0..*count.get_untracked() {
|
2022-10-21 14:57:50 -04:00
|
|
|
acc.accumulate_from(
|
|
|
|
self.recipes
|
|
|
|
.get()
|
|
|
|
.get(idx)
|
|
|
|
.expect(&format!("No such recipe id exists: {}", idx)),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if show_staples {
|
|
|
|
if let Some(staples) = self.staples.get().as_ref() {
|
|
|
|
acc.accumulate_from(staples);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let mut ingredients = acc.ingredients();
|
|
|
|
let mut groups = BTreeMap::new();
|
|
|
|
let cat_map = self.category_map.get().clone();
|
|
|
|
for (_, (i, recipes)) in ingredients.iter_mut() {
|
|
|
|
let category = if let Some(cat) = cat_map.get(&i.name) {
|
|
|
|
cat.clone()
|
|
|
|
} else {
|
|
|
|
"other".to_owned()
|
|
|
|
};
|
|
|
|
i.category = category.clone();
|
|
|
|
groups
|
|
|
|
.entry(category)
|
|
|
|
.or_insert(vec![])
|
|
|
|
.push((i.clone(), recipes.clone()));
|
|
|
|
}
|
|
|
|
debug!(?self.category_map);
|
|
|
|
// FIXME(jwall): Sort by categories and names.
|
|
|
|
groups
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:45:14 -05:00
|
|
|
/// Retrieves the count for a recipe without triggering subscribers to the entire
|
|
|
|
/// recipe count set.
|
|
|
|
pub fn get_recipe_count_by_index(&self, key: &String) -> Option<RcSignal<usize>> {
|
|
|
|
self.recipe_counts.get_untracked().get(key).cloned()
|
2022-10-21 14:57:50 -04:00
|
|
|
}
|
|
|
|
|
2022-11-11 16:45:14 -05:00
|
|
|
pub fn reset_recipe_counts(&self) {
|
2022-11-11 17:35:10 -05:00
|
|
|
for (_, count) in self.recipe_counts.get_untracked().iter() {
|
2022-11-11 16:45:14 -05:00
|
|
|
count.set(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Set the recipe_count by index. Does not trigger subscribers to the entire set of recipe_counts.
|
|
|
|
/// This does trigger subscribers of the specific recipe you are updating though.
|
|
|
|
pub fn set_recipe_count_by_index(&self, key: &String, count: usize) -> RcSignal<usize> {
|
|
|
|
let mut counts = self.recipe_counts.get_untracked().as_ref().clone();
|
|
|
|
counts
|
|
|
|
.entry(key.clone())
|
|
|
|
.and_modify(|e| e.set(count))
|
|
|
|
.or_insert_with(|| create_rc_signal(count));
|
2022-10-21 14:57:50 -04:00
|
|
|
self.recipe_counts.set(counts);
|
2022-11-11 16:45:14 -05:00
|
|
|
self.recipe_counts.get_untracked().get(key).unwrap().clone()
|
2022-10-21 14:57:50 -04:00
|
|
|
}
|
2022-11-11 17:35:10 -05:00
|
|
|
|
|
|
|
pub fn get_current_modified_amts(&self) -> BTreeMap<IngredientKey, String> {
|
|
|
|
let mut modified_amts = BTreeMap::new();
|
|
|
|
for (key, amt) in self.modified_amts.get_untracked().iter() {
|
|
|
|
modified_amts.insert(key.clone(), amt.get_untracked().as_ref().clone());
|
|
|
|
}
|
|
|
|
modified_amts
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reset_modified_amts(&self, modified_amts: BTreeMap<IngredientKey, String>) {
|
|
|
|
let mut modified_amts_copy = self.modified_amts.get().as_ref().clone();
|
|
|
|
for (key, amt) in modified_amts {
|
|
|
|
modified_amts_copy
|
|
|
|
.entry(key)
|
|
|
|
.and_modify(|amt_signal| amt_signal.set(amt.clone()))
|
|
|
|
.or_insert_with(|| create_rc_signal(amt));
|
|
|
|
}
|
|
|
|
self.modified_amts.set(modified_amts_copy);
|
|
|
|
}
|
2022-10-21 14:57:50 -04:00
|
|
|
}
|