diff --git a/crates/frontend/src/components/condition_pills.rs b/crates/frontend/src/components/condition_pills.rs index b39e45be..e75376ed 100644 --- a/crates/frontend/src/components/condition_pills.rs +++ b/crates/frontend/src/components/condition_pills.rs @@ -1,15 +1,12 @@ -pub mod types; -pub mod utils; +use crate::{ + logic::{Condition, Conditions, Expression, Operator}, + schema::HtmlDisplay, +}; use leptos::{leptos_dom::helpers::WindowListenerHandle, *}; -use serde_json::Value; use wasm_bindgen::JsCast; use web_sys::Element; -use crate::components::condition_pills::types::ConditionOperator; - -use self::types::Condition; - use derive_more::{Deref, DerefMut}; #[derive(Debug, Clone, Deref, DerefMut, Default)] @@ -60,33 +57,14 @@ pub fn condition_expression( } else { ("condition-item-collapsed", "condition-value-collapsed") }; - let Condition { left_operand: dimension, operator, right_operand: value } = condition - .get_value(); - let filtered_vals: Vec = value - .into_iter() - .filter_map(|v| { - if v.is_object() && v.get("var").is_some() { - None - } else { - match v { - Value::String(s) => Some(s.to_string()), - Value::Number(n) => Some(n.to_string()), - Value::Bool(b) => Some(b.to_string()), - Value::Array(arr) => { - Some( - arr - .iter() - .map(|v| v.to_string()) - .collect::>() - .join(","), - ) - } - Value::Object(o) => serde_json::to_string_pretty(&o).ok(), - _ => None, - } - } - }) - .collect(); + let (dimension, operator, operands): (String, Operator, Vec) = condition + .with_value(|v| { + ( + v.variable.clone(), + <&Condition as Into>::into(v), + v.expression.to_constants_vec().iter().map(|c| c.html_display()).collect(), + ) + }); view! {
  • - {match operator { - ConditionOperator::Between => { - if filtered_vals.len() == 2 { - view! { - <> - - {&filtered_vals[0]} - - - {"and"} - - - {&filtered_vals[1]} - - - } - .into_view() - } else { - view! { - - "Invalid between values" + {match condition.get_value().expression { + Expression::Between(c1, c2) => { + view! { + <> + + {c1.html_display()} + + + {"and"} - } - .into_view() + + {c2.html_display()} + + } + .into_view() } _ => { - let rendered_value = filtered_vals.join(", "); + let rendered_value = operands.join(", "); view! { {rendered_value} }.into_view() } }} @@ -146,7 +115,7 @@ pub fn condition_expression( #[component] pub fn condition( #[prop(into)] id: String, - #[prop(into)] conditions: Vec, + #[prop(into)] conditions: Conditions, #[prop(into, default=String::new())] class: String, #[prop(default = true)] grouped_view: bool, ) -> impl IntoView { @@ -164,11 +133,17 @@ pub fn condition( .clone()> {conditions .get_value() - .into_iter() + .iter() .enumerate() .map(|(idx, condition)| { let item_id = format!("{}-{}", id, idx); - view! { } + view! { + + } }) .collect::>()} diff --git a/crates/frontend/src/components/condition_pills/types.rs b/crates/frontend/src/components/condition_pills/types.rs deleted file mode 100644 index 9bff2921..00000000 --- a/crates/frontend/src/components/condition_pills/types.rs +++ /dev/null @@ -1,138 +0,0 @@ -use std::fmt::Display; - -use serde::{Deserialize, Serialize}; -use serde_json::{Map, Value}; -use superposition_types::Context; - -#[derive(Debug, Clone, Serialize, Deserialize)] -pub enum ConditionOperator { - Is, - In, - Has, - Between, - Other(String), -} - -impl Display for ConditionOperator { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - match self { - Self::Has => f.write_str("has"), - Self::Is => f.write_str("is"), - Self::In => f.write_str("in"), - Self::Between => f.write_str("between"), - Self::Other(o) => f.write_str(o), - } - } -} - -impl From for ConditionOperator { - fn from(op: String) -> Self { - match op.as_str() { - "==" => ConditionOperator::Is, - "<=" => ConditionOperator::Between, - "in" => ConditionOperator::In, - "has" => ConditionOperator::Has, - other => ConditionOperator::Other(other.to_string()), - } - } -} - -impl From<(String, &Vec)> for ConditionOperator { - fn from(value: (String, &Vec)) -> Self { - let (operator, operands) = value; - let operand_0 = operands.first(); - let operand_1 = operands.get(1); - let operand_2 = operands.get(2); - match (operator.as_str(), operand_0, operand_1, operand_2) { - // assuming there will be only two operands, one with the dimension name and other with the value - ("==", _, _, None) => ConditionOperator::Is, - ("<=", Some(_), Some(Value::Object(a)), Some(_)) if a.contains_key("var") => { - ConditionOperator::Between - } - // assuming there will be only two operands, one with the dimension name and other with the value - ("in", Some(Value::Object(a)), Some(_), None) if a.contains_key("var") => { - ConditionOperator::In - } - // assuming there will be only two operands, one with the dimension name and other with the value - ("in", Some(_), Some(Value::Object(a)), None) if a.contains_key("var") => { - ConditionOperator::Has - } - _ => ConditionOperator::Other(operator), - } - } -} - -#[derive(Clone, Debug, Serialize, Deserialize)] -pub struct Condition { - pub left_operand: String, - pub operator: ConditionOperator, - pub right_operand: Vec, -} - -#[derive(Default, Clone)] -pub struct Conditions(pub Vec); - -impl TryFrom<&Map> for Condition { - type Error = &'static str; - fn try_from(source: &Map) -> Result { - if let Some(operator) = source.keys().next() { - let emty_vec = vec![]; - let operands = source[operator].as_array().unwrap_or(&emty_vec); - - let operator = ConditionOperator::from((operator.to_owned(), operands)); - - let dimension_name = operands - .iter() - .find_map(|item| match item.as_object() { - Some(o) if o.contains_key("var") => { - Some(o["var"].as_str().unwrap_or("")) - } - _ => None, - }) - .unwrap_or(""); - - return Ok(Condition { - operator, - left_operand: dimension_name.to_owned(), - right_operand: operands - .to_vec() - .into_iter() - .filter(|operand| { - !(operand.is_object() && operand.get("var").is_some()) - }) - .collect(), - }); - } - - Err("not a valid condition map") - } -} - -impl TryFrom<&Value> for Condition { - type Error = &'static str; - fn try_from(value: &Value) -> Result { - let obj = value - .as_object() - .ok_or("not a valid condition value, should be an object")?; - Condition::try_from(obj) - } -} - -impl TryFrom<&Context> for Conditions { - type Error = &'static str; - fn try_from(context: &Context) -> Result { - let obj: Map = context.condition.clone().into(); - match obj.get("and") { - Some(v) => v - .as_array() - .ok_or("failed to parse value of and as array") - .and_then(|arr| { - arr.iter() - .map(Condition::try_from) - .collect::, &'static str>>() - }), - None => Condition::try_from(&obj).map(|v| vec![v]), - } - .map(Self) - } -} diff --git a/crates/frontend/src/components/condition_pills/utils.rs b/crates/frontend/src/components/condition_pills/utils.rs deleted file mode 100644 index 53f3092c..00000000 --- a/crates/frontend/src/components/condition_pills/utils.rs +++ /dev/null @@ -1,18 +0,0 @@ -use super::types::Condition; -use serde_json::Value; - -pub fn extract_conditions(context: &Value) -> Vec { - context - .as_object() - .and_then(|obj| { - obj.get("and") - .and_then(|v| v.as_array()) - .map(|arr| { - arr.iter() - .filter_map(|condition| Condition::try_from(condition).ok()) - .collect::>() - }) - .or_else(|| Condition::try_from(obj).ok().map(|v| vec![v])) - }) - .unwrap_or_default() -} diff --git a/crates/frontend/src/components/context_card.rs b/crates/frontend/src/components/context_card.rs index 36a7e9d4..7094e1e1 100644 --- a/crates/frontend/src/components/context_card.rs +++ b/crates/frontend/src/components/context_card.rs @@ -2,10 +2,12 @@ use leptos::*; use serde_json::{Map, Value}; use superposition_types::Context; -use crate::components::condition_pills::types::{ConditionOperator, Conditions}; -use crate::components::{ - condition_pills::Condition as ConditionComponent, - table::{types::Column, Table}, +use crate::{ + components::{ + condition_pills::Condition as ConditionComponent, + table::{types::Column, Table}, + }, + logic::{Conditions, Expression}, }; #[component] @@ -46,16 +48,13 @@ pub fn context_card( Column::default("VALUE".to_string()), ]; - let actions_supported = show_actions - && !conditions - .0 - .iter() - .any(|condition| condition.left_operand == "variantIds"); + let actions_supported = + show_actions && !conditions.0.iter().any(|c| c.variable == "variantIds"); let edit_unsupported = conditions .0 .iter() - .any(|condition| matches!(condition.operator, ConditionOperator::Other(_))); + .any(|c| matches!(c.expression, Expression::Other(_, _))); view! {
    @@ -109,7 +108,7 @@ pub fn context_card(
    diff --git a/crates/frontend/src/components/context_form.rs b/crates/frontend/src/components/context_form.rs index f762f823..fa80e4b5 100644 --- a/crates/frontend/src/components/context_form.rs +++ b/crates/frontend/src/components/context_form.rs @@ -2,470 +2,407 @@ pub mod utils; use std::collections::{HashMap, HashSet}; +use crate::components::input::{Input, InputType}; +use crate::logic::{Condition, Conditions, Expression, Operator}; +use crate::schema::EnumVariants; +use crate::{ + components::dropdown::{Dropdown, DropdownDirection}, + schema::SchemaType, +}; use leptos::*; -use serde_json::{Map, Value}; +use serde_json::Value; use superposition_types::database::types::DimensionWithMandatory; -use web_sys::MouseEvent; -use crate::components::{ - condition_pills::types::ConditionOperator, - dropdown::{Dropdown, DropdownDirection}, - input_components::{BooleanToggle, EnumDropdown}, -}; -use crate::utils::get_key_type; +#[component] +pub fn condition_input( + disabled: bool, + resolve_mode: bool, + allow_remove: bool, + condition: StoredValue, + input_type: StoredValue, + schema_type: StoredValue, + #[prop(into)] on_remove: Callback, + #[prop(into)] on_value_change: Callback, + #[prop(into)] on_operator_change: Callback, +) -> impl IntoView { + let (dimension, operator): (String, Operator) = + condition.with_value(|v| (v.variable.clone(), v.into())); + + let inputs: Vec<(Value, Callback)> = match condition.get_value().expression + { + Expression::Is(c) => { + vec![( + c, + Callback::new(move |value: Value| { + on_value_change.call(Expression::Is(value)); + }), + )] + } + Expression::In(c) => { + vec![( + c, + Callback::new(move |value: Value| { + on_value_change.call(Expression::In(value)); + }), + )] + } + Expression::Has(c) => { + vec![( + c, + Callback::new(move |value: Value| { + on_value_change.call(Expression::Has(value)); + }), + )] + } + Expression::Between(c1, c2) => { + let c2_clone = c2.clone(); + vec![ + ( + c1.clone(), + Callback::new(move |value: Value| { + on_value_change + .call(Expression::Between(value, c2_clone.clone())); + }), + ), + ( + c2.clone(), + Callback::new(move |value: Value| { + on_value_change.call(Expression::Between(c1.clone(), value)); + }), + ), + ] + } + _ => vec![], + }; + view! { +
    +
    + + +
    +
    + + + -use super::condition_pills::types::Condition; +
    +
    +
    + + +
    + + {inputs + .into_iter() + .enumerate() + .map(|(idx, (value, on_change)): (usize, (Value, Callback))| { + view! { + >::into(v)), + ) + }), + idx, + ) + + class="w-[450px]" + name="" + operator=Some(operator.clone()) + /> + } + }) + .collect_view()} + + +
    +
    + } +} #[component] pub fn context_form( handle_change: NF, + context: Conditions, dimensions: Vec, - #[prop(default = false)] is_standalone: bool, - context: Vec, - #[prop(default = String::new())] heading_sub_text: String, #[prop(default = false)] disabled: bool, - #[prop(default = DropdownDirection::Right)] dropdown_direction: DropdownDirection, #[prop(default = false)] resolve_mode: bool, + #[prop(default = String::new())] heading_sub_text: String, + #[prop(default = DropdownDirection::Right)] dropdown_direction: DropdownDirection, ) -> impl IntoView where - NF: Fn(Vec) + 'static, + NF: Fn(Conditions) + 'static, { - // let _has_dimensions = !dimensions.is_empty(); - - let (used_dimensions, set_used_dimensions) = create_signal( + let dimension_map = store_value( + dimensions + .iter() + .map(|v| (v.dimension.clone(), v.clone())) + .collect::>(), + ); + let (used_dimensions_rs, used_dimensions_ws) = create_signal( context .iter() - .map(|condition| condition.left_operand.clone()) + .map(|condition| condition.variable.clone()) .collect::>(), ); - let (context, set_context) = create_signal(context.clone()); + let (context_rs, context_ws) = create_signal(context.clone()); let dimensions = StoredValue::new(dimensions); let mandatory_dimensions = StoredValue::new( dimensions .get_value() .into_iter() - .filter_map(|dim| { - if dim.mandatory { - Some(dim.dimension) - } else { - None - } - }) + .filter(|dim| dim.mandatory) + .map(|dim| dim.dimension) .collect::>(), ); - let last_idx = create_memo(move |_| context.get().len().max(1) - 1); - - let on_click = move |event: MouseEvent| { - event.prevent_default(); - logging::log!("Context form submit"); - //TODO: submit logic for this - }; + let last_idx = create_memo(move |_| context_rs.get().len().max(1) - 1); create_effect(move |_| { - let f_context = context.get(); // context will now be a Value + let f_context = context_rs.get(); // context will now be a Value logging::log!("Context form effect {:?}", f_context); handle_change(f_context.clone()); // handle_change now expects Value }); - let handle_select_dropdown_option = + let on_select_dimension = Callback::new(move |selected_dimension: DimensionWithMandatory| { let dimension_name = selected_dimension.dimension; - set_used_dimensions.update(|value: &mut HashSet| { - value.insert(dimension_name.clone()); - }); - set_context.update(|value| { - value.push(Condition { - left_operand: dimension_name.clone(), - operator: ConditionOperator::Is, - right_operand: vec![Value::String("".to_string())], - }) - }); - }); - - view! { -
    -
    -
    - -
    -
    -
    - -
    - -
    -
    - {move || { - let dimensions_map = dimensions - .get_value() - .into_iter() - .map(|ele| (ele.dimension.clone(), ele)) - .collect::>(); - view! { - >() - } - - key=|(idx, condition)| { - format!("{}-{}-{}", condition.left_operand, idx, condition.operator) - } - - children=move |(idx, condition)| { - let dimension_label = condition.left_operand.to_string(); - // let dimension_label = dimension.to_string(); - let dimension_name = StoredValue::new( - condition.left_operand.to_string(), - ); - let schema: Map = serde_json::from_value( - dimensions_map.get(&dimension_label).unwrap().schema.clone(), - ) - .unwrap(); - let dimension_type = get_key_type(&schema); - if let ConditionOperator::Other(ref op_str) = condition.operator { - if op_str.is_empty() { - set_context.update_untracked(|curr_context| { - curr_context[idx].operator = ConditionOperator::Is; - }); - let mut_operator = String::from("=="); - set_context.update_untracked(|curr_context| { - curr_context[idx].operator = ConditionOperator::Other(mut_operator.clone()); - }); - } - } - view! { - // - -
    -
    - - -
    -
    - - - - -
    -
    - -
    + if let Ok(r#type) = SchemaType::try_from(selected_dimension.schema) { + used_dimensions_ws.update(|value: &mut HashSet| { + value.insert(dimension_name.clone()); + }); + context_ws.update(|value| { + value.push(Condition::new_with_default_expression( + dimension_name, + r#type, + )) + }); + } + // TODO show alert in case of invalid dimension + }); - { - // Generate input fields based on the condition's right_operand (Vec) - let input_fields = match &condition.right_operand { - values => { - // Filter out any elements that are objects containing a "var" key - let filtered_elements: Vec<_> = values - .iter() - .filter(|v| !v.is_object() || !v.get("var").is_some()) // Exclude elements with "var" - .collect(); + let on_operator_change = + Callback::new(move |(idx, expression): (usize, Expression)| { + context_ws.update(|v| { + if idx < v.len() { + v[idx].expression = expression; + } + }); + }); - // Directly return the input fields - filtered_elements - .into_iter() // Use `into_iter` to consume the filtered_elements - .enumerate() - .map(|(i, element)| match element { - Value::String(s) => view! { - - }.into_view(), + let on_value_change = Callback::new(move |(idx, expression): (usize, Expression)| { + context_ws.update(|v| { + if idx < v.len() { + v[idx].expression = expression; + } + }) + }); - Value::Number(n) => view! { - () { // Try to parse input as f64 - set_context.update(|curr_context| { - if let Some(elem) = curr_context[idx].right_operand.get_mut(i) { - if !elem.is_object() || !elem.get("var").is_some() { // Exclude elements with "var" - *elem = Value::Number(serde_json::Number::from_f64(parsed).unwrap()); - } - } - }); - } - } - name="context-dimension-value" - type="number" - placeholder="Type here" - class="input input-bordered w-full bg-white text-gray-700 shadow-md" - /> - }.into_view(), + let on_remove = Callback::new(move |(idx, d_name): (usize, String)| { + used_dimensions_ws.update(|value| { + value.remove(&d_name); + }); + context_ws.update(|v| { + v.remove(idx); + }); + }); - Value::Bool(b) => view! { - +
    + +
    +
    +
    + +
    + - }.into_view(), +
    +
    + >() + } - _ => view! { - - }.into_view(), - }) - .collect::>() // Collect the result into a Vec - } - }; - match condition.operator { - ConditionOperator::Is => { - match dimension_type.as_str() { - "ENUM" => { - let filtered_value = condition.right_operand - .iter() - .find(|v| match v { - Value::Object(obj) => !obj.contains_key("var"), - _ => true, - }) - .map(|v| v.to_string().replace('"', "")) - .unwrap_or_else(|| String::new()); + key=|(idx, condition)| { + format!( + "{}-{}-{}", + condition.variable, + idx, + condition.expression.to_operator(), + ) + } - view! { - - } - .into_view() - } - "BOOLEAN" => { - // Ensure we handle Value properly and default to false if not a valid boolean - let is_checked = match &condition.right_operand[0] { - Value::Bool(b) => *b, // Extract the boolean value directly - _ => false, // Default to false if not a boolean - }; - view! { - - } - .into_view() - } - _ => view! { - { - logging::log!("Condition operator and saurav {:?}", condition.operator); - input_fields.into_view() - } - } // Fallback to input field if not ENUM or BOOLEAN - } - } - _ => view! { - {input_fields.into_view()} - } // For other operators, use input_fields as default - } - } - - - -
    -
    -
    + +
    - {move || { - if last_idx.get() != idx { - view! { -
    - "&&" -
    - } - .into_view() - } else { - view! {}.into_view() - } - }} - } - } + {move || { + let dimensions = dimensions + .get_value() + .into_iter() + .filter(|dimension| { + !used_dimensions_rs.get().contains(&dimension.dimension) + }) + .collect::>(); + view! { + } }} - -
    - - {move || { - let dimensions = dimensions - .get_value() - .into_iter() - .filter(|dimension| { - !used_dimensions.get().contains(&dimension.dimension) - }) - .collect::>(); - view! { - - } - }} - -
    -
    -
    -
    + +
    - -
    - -
    -
    - } +
    + } } diff --git a/crates/frontend/src/components/context_form/utils.rs b/crates/frontend/src/components/context_form/utils.rs index ef252443..02e31bc4 100644 --- a/crates/frontend/src/components/context_form/utils.rs +++ b/crates/frontend/src/components/context_form/utils.rs @@ -1,241 +1,27 @@ +use super::Conditions; +use crate::utils::{construct_request_headers, get_host, parse_json_response, request}; + use anyhow::Result; -use leptos::logging; use serde_json::{json, Map, Value}; -use superposition_types::database::types::DimensionWithMandatory; - -use crate::{ - components::condition_pills::types::{Condition, ConditionOperator}, - utils::{ - construct_request_headers, get_config_value, get_host, parse_json_response, - request, ConfigType, - }, -}; - -pub fn get_condition_schema( - condition: &Condition, - dimensions: Vec, -) -> Result { - let var = &condition.left_operand; // Dimension name - let op = &condition.operator; // Operator type - let val = &condition.right_operand; // Vec - - // Extract non-"var" elements from the right_operand - let filtered_values: Vec<&Value> = val - .iter() - .filter(|v| !v.is_object() || !v.get("var").is_some()) // Ignore objects with "var" - .collect(); - let dimensions_clone = dimensions.clone(); - - match op { - ConditionOperator::Between => { - // Expecting three elements for "Between" condition: two operands and one "var" object - if filtered_values.len() != 2 { - return Err( - "Invalid number of operands for 'between' condition.".to_string() - ); - } - - let first_operand = &filtered_values[0]; // The first value - let third_operand = &filtered_values[1]; // The third value - - let first_operand_value = get_config_value( - var, - first_operand, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - - let third_operand_value = get_config_value( - var, - third_operand, - &dimensions_clone - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - - Ok(json!({ - "<=": [ - first_operand_value, - { "var": var }, - third_operand_value - ] - })) - } - ConditionOperator::Is => { - // Expecting two elements for "Is" condition: one "var" object and one value - if filtered_values.len() != 1 { - return Err("Invalid number of operands for 'is' condition.".to_string()); - } - - let value = &filtered_values[0]; // The value after "var" - let first_operand_value = get_config_value( - var, - value, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - - Ok(json!({ - "==": [ - { "var": var }, - first_operand_value - ] - })) - } - ConditionOperator::In => { - if filtered_values.len() != 1 { - return Err("Invalid number of operands for 'in' condition.".to_string()); - } - let value = &filtered_values[0]; // The value after "var" - let first_operand_value = get_config_value( - var, - value, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - Ok(json!({ - "in": [ - { "var": var }, - first_operand_value - ] - })) - } - ConditionOperator::Has => { - if filtered_values.len() != 1 { - return Err("Invalid number of operands for 'has' condition.".to_string()); - } - let value = &filtered_values[0]; // The value after "var" - let first_operand_value = get_config_value( - var, - value, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - - Ok(json!({ - "in": [ - first_operand_value, - { "var": var } - ] - })) - } - ConditionOperator::Other(op) => { - if filtered_values.len() == 1 { - let value = &filtered_values[0]; // The value after "var" - let first_operand_value = get_config_value( - var, - value, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - Ok(json!({ - op: [ - { "var": var }, - first_operand_value - ] - })) - } else if filtered_values.len() == 2 { - let first_operand = &filtered_values[0]; // The first value - let second_operand = &filtered_values[1]; // The second value - let first_operand_value = get_config_value( - var, - first_operand, - &dimensions - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - let second_operand_value = get_config_value( - var, - second_operand, - &dimensions_clone - .into_iter() - .map(ConfigType::Dimension) - .collect::>(), - )?; - Ok(json!({ - op: [ - first_operand_value, - { "var": var }, - second_operand_value - ] - })) - } else { - Err("Invalid number of operands for custom operator.".to_string()) - } - } - } -} - -pub fn construct_context( - conditions: Vec, - dimensions: Vec, -) -> Value { - if conditions.is_empty() { - json!({}) - } else { - let condition_schemas = conditions - .iter() - .map( - |condition| match get_condition_schema(condition, dimensions.clone()) { - Ok(ans) => ans, - Err(err) => { - logging::log!("Error while getting condition schema {:?}", err); - Value::Null - } - }, - ) - .collect::>(); - - if condition_schemas.len() == 1 { - condition_schemas[0].clone() - } else { - json!({ "and": condition_schemas }) - } - } -} - -pub fn construct_request_payload( - overrides: Map, - conditions: Vec, - dimensions: Vec, -) -> Value { - // Construct the override section - let override_section: Map = overrides; - - // Construct the context section - let context_section = construct_context(conditions, dimensions.clone()); - - // Construct the entire request payload - let request_payload = json!({ - "override": override_section, - "context": context_section +pub fn context_payload(overrides: Map, conditions: Conditions) -> Value { + let context: Value = conditions.to_context_json(); + let payload = json!({ + "override": overrides, + "context": context }); - request_payload + payload } pub async fn create_context( tenant: String, overrides: Map, - conditions: Vec, - dimensions: Vec, -) -> Result { + conditions: Conditions, +) -> Result { let host = get_host(); let url = format!("{host}/context"); - let request_payload = construct_request_payload(overrides, conditions, dimensions); + let request_payload = context_payload(overrides, conditions); let response = request( url, reqwest::Method::PUT, @@ -250,13 +36,11 @@ pub async fn create_context( pub async fn update_context( tenant: String, overrides: Map, - conditions: Vec, - dimensions: Vec, + conditions: Conditions, ) -> Result { let host = get_host(); let url = format!("{host}/context/overrides"); - let request_payload = - construct_request_payload(overrides, conditions, dimensions.clone()); + let request_payload = context_payload(overrides, conditions); let response = request( url, reqwest::Method::PUT, diff --git a/crates/frontend/src/components/experiment.rs b/crates/frontend/src/components/experiment.rs index 9642da4e..d24d5b6e 100644 --- a/crates/frontend/src/components/experiment.rs +++ b/crates/frontend/src/components/experiment.rs @@ -8,6 +8,7 @@ use superposition_types::database::models::experimentation::{ use crate::components::table::types::Column; use crate::components::table::Table; + use crate::schema::HtmlDisplay; use crate::types::Experiment; @@ -192,35 +193,29 @@ where

    Context

    - {move || { - let mut view = Vec::new(); - for token in contexts.clone() { - let (dimension, values) = (token.left_operand, token.right_operand); - let mut value_views = Vec::new(); - for value in values.iter() { - if value.is_object() && value.get("var").is_some() { - continue; - } - value_views - .push( - view! { -
    - {value.html_display()} -
    - }, - ); + + {contexts + .iter() + .map(|condition| { + let dimension = condition.variable.clone(); + let operand_views = condition + .expression + .to_constants_vec() + .iter() + .map(|c| { + view! { +
    {c.html_display()}
    + } + }) + .collect_view(); + view! { +
    +
    {dimension}
    + {operand_views} +
    } - view.push( - view! { -
    -
    {dimension}
    - {value_views} -
    - }, - ); - } - view - }} + }) + .collect_view()}
    diff --git a/crates/frontend/src/components/experiment_conclude_form/utils.rs b/crates/frontend/src/components/experiment_conclude_form/utils.rs index b73829c9..7113a55a 100644 --- a/crates/frontend/src/components/experiment_conclude_form/utils.rs +++ b/crates/frontend/src/components/experiment_conclude_form/utils.rs @@ -1,13 +1,13 @@ use leptos::logging::log; use serde_json::json; -use crate::{types::Experiment, utils::get_host}; +use crate::{types::ExperimentResponse, utils::get_host}; pub async fn conclude_experiment( exp_id: String, variant_id: String, tenant: &String, -) -> Result { +) -> Result { let client = reqwest::Client::new(); let host = get_host(); match client @@ -20,7 +20,7 @@ pub async fn conclude_experiment( Ok(experiment) => { log!("experiment response {:?}", experiment); Ok(experiment - .json::() + .json::() .await .map_err(|err| err.to_string())?) } diff --git a/crates/frontend/src/components/experiment_form.rs b/crates/frontend/src/components/experiment_form.rs index 0f173c7e..4de5830f 100644 --- a/crates/frontend/src/components/experiment_form.rs +++ b/crates/frontend/src/components/experiment_form.rs @@ -14,7 +14,7 @@ use crate::components::context_form::ContextForm; use crate::components::variant_form::VariantForm; use crate::types::{VariantFormT, VariantFormTs}; -use super::condition_pills::types::Condition; +use crate::logic::Conditions; fn default_variants_for_form() -> Vec<(String, VariantFormT)> { vec![ @@ -55,7 +55,7 @@ pub fn experiment_form( #[prop(default = false)] edit: bool, #[prop(default = String::new())] id: String, name: String, - context: Vec, + context: Conditions, variants: VariantFormTs, handle_submit: NF, default_config: Vec, @@ -73,7 +73,7 @@ where let (f_variants, set_variants) = create_signal(init_variants); let (req_inprogess_rs, req_inprogress_ws) = create_signal(false); - let handle_context_form_change = move |updated_ctx: Vec| { + let handle_context_form_change = move |updated_ctx: Conditions| { set_context.set_untracked(updated_ctx); }; @@ -108,14 +108,8 @@ where let result = if edit { update_experiment(experiment_id, f_variants, tenant).await } else { - create_experiment( - f_context, - f_variants, - f_experiment_name, - tenant, - dimensions.get_value().clone(), - ) - .await + create_experiment(f_context, f_variants, f_experiment_name, tenant) + .await }; match result { @@ -157,11 +151,9 @@ where let context = f_context.get(); view! { dimensions=dimensions.get_value() context=context handle_change=handle_context_form_change - is_standalone=false disabled=edit heading_sub_text=String::from( "Define rules under which this experiment would run", diff --git a/crates/frontend/src/components/experiment_form/utils.rs b/crates/frontend/src/components/experiment_form/utils.rs index 27ecdbe2..29cfaa53 100644 --- a/crates/frontend/src/components/experiment_form/utils.rs +++ b/crates/frontend/src/components/experiment_form/utils.rs @@ -1,13 +1,8 @@ -use serde_json::Value; -use superposition_types::database::types::DimensionWithMandatory; - -use crate::components::condition_pills::types::Condition; -use crate::components::context_form::utils::construct_context; +use super::types::{ExperimentCreateRequest, ExperimentUpdateRequest}; +use crate::logic::Conditions; use crate::types::VariantFormT; use crate::utils::{construct_request_headers, get_host, parse_json_response, request}; -use super::types::{ExperimentCreateRequest, ExperimentUpdateRequest}; - pub fn validate_experiment(experiment: &ExperimentCreateRequest) -> Result { if experiment.name.is_empty() { return Err(String::from("experiment name should not be empty")); @@ -16,16 +11,15 @@ pub fn validate_experiment(experiment: &ExperimentCreateRequest) -> Result, + conditions: Conditions, variants: Vec, name: String, tenant: String, - dimensions: Vec, -) -> Result { +) -> Result { let payload = ExperimentCreateRequest { name, variants: FromIterator::from_iter(variants), - context: construct_context(conditions, dimensions.clone()), + context: conditions.to_context_json(), }; let _ = validate_experiment(&payload)?; @@ -47,7 +41,7 @@ pub async fn update_experiment( experiment_id: String, variants: Vec, tenant: String, -) -> Result { +) -> Result { let payload = ExperimentUpdateRequest { variants: FromIterator::from_iter(variants), }; diff --git a/crates/frontend/src/components/experiment_ramp_form/utils.rs b/crates/frontend/src/components/experiment_ramp_form/utils.rs index 7018ad61..9722c7be 100644 --- a/crates/frontend/src/components/experiment_ramp_form/utils.rs +++ b/crates/frontend/src/components/experiment_ramp_form/utils.rs @@ -1,13 +1,13 @@ use leptos::logging::log; use serde_json::json; -use crate::{types::Experiment, utils::get_host}; +use crate::{types::ExperimentResponse, utils::get_host}; pub async fn ramp_experiment( exp_id: &String, percent: u8, tenant: &String, -) -> Result { +) -> Result { let client = reqwest::Client::new(); let host = get_host(); match client @@ -20,7 +20,7 @@ pub async fn ramp_experiment( Ok(experiment) => { log!("experiment response {:?}", experiment); Ok(experiment - .json::() + .json::() .await .map_err(|err| err.to_string())?) } diff --git a/crates/frontend/src/components/override_form.rs b/crates/frontend/src/components/override_form.rs index faaa0ed0..1c42041a 100644 --- a/crates/frontend/src/components/override_form.rs +++ b/crates/frontend/src/components/override_form.rs @@ -3,7 +3,6 @@ use std::collections::{HashMap, HashSet}; use leptos::*; use serde_json::Value; use superposition_types::database::models::cac::DefaultConfig; -use web_sys::MouseEvent; use crate::{ components::{ @@ -63,7 +62,7 @@ fn override_input(
    @@ -85,8 +84,7 @@ fn override_input( .into_view() } else { view! {

    "An Error Occured"

    }.into_view() - }} - + }}
    - -
    - -
    -
    } } diff --git a/crates/frontend/src/components/variant_form.rs b/crates/frontend/src/components/variant_form.rs index e51940ee..c4deade3 100644 --- a/crates/frontend/src/components/variant_form.rs +++ b/crates/frontend/src/components/variant_form.rs @@ -297,7 +297,6 @@ where overrides=overrides default_config=default_config.get_value() handle_change=handle_change - is_standalone=false show_add_override=false handle_key_remove=Some(Callback::new(on_key_remove)) /> @@ -309,7 +308,6 @@ where overrides=overrides default_config=default_config.get_value() handle_change=handle_change - is_standalone=false show_add_override=false disable_remove=true /> diff --git a/crates/frontend/src/logic.rs b/crates/frontend/src/logic.rs index 891e7c8c..7366eee5 100644 --- a/crates/frontend/src/logic.rs +++ b/crates/frontend/src/logic.rs @@ -6,6 +6,11 @@ use serde_json::{json, Map}; use crate::schema::{HtmlDisplay, SchemaType}; use superposition_types::Context; +/// The `Expression` enum represents a JSONLogic expression and ensures +/// the correct construction of expressions by tying operators and operands together. +/// +/// This enum provides a way to enforce the proper order and number of operands +/// for each operator, ensuring the resulting JSONLogic expression is well-formed. #[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] pub enum Expression { Is(serde_json::Value), @@ -16,7 +21,10 @@ pub enum Expression { } pub fn is_variable(v: &serde_json::Value) -> bool { - v.is_object() && v.as_object().unwrap().contains_key("var") + v.is_object() + && v.as_object() + .map(|v| v.contains_key("var")) + .unwrap_or_default() } pub fn is_constant(v: &serde_json::Value) -> bool { @@ -33,7 +41,7 @@ impl Expression { .cloned() .ok_or("Invalid operands list for context")?; - let operand_0 = operands.first(); + let operand_0 = operands.get(0); let operand_1 = operands.get(1); let operand_2 = operands.get(2); @@ -176,6 +184,8 @@ impl From<(SchemaType, Operator)> for Expression { } } +/// This a copy of `Expression` enum, to prevent copying/moving of data +/// where just the operator information is needed. #[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] pub enum Operator { Is, diff --git a/crates/frontend/src/pages/context_override.rs b/crates/frontend/src/pages/context_override.rs index 5d3057a2..2263e3e6 100644 --- a/crates/frontend/src/pages/context_override.rs +++ b/crates/frontend/src/pages/context_override.rs @@ -11,8 +11,6 @@ use superposition_types::{ use crate::api::{delete_context, fetch_config, fetch_default_config, fetch_dimensions}; use crate::components::alert::AlertType; use crate::components::button::Button; -use crate::components::condition_pills::types::{Condition, ConditionOperator}; -use crate::components::condition_pills::utils::extract_conditions; use crate::components::context_card::ContextCard; use crate::components::context_form::utils::{create_context, update_context}; use crate::components::context_form::ContextForm; @@ -20,13 +18,15 @@ use crate::components::delete_modal::DeleteModal; use crate::components::drawer::{close_drawer, open_drawer, Drawer, DrawerBtn}; use crate::components::override_form::OverrideForm; use crate::components::skeleton::{Skeleton, SkeletonVariant}; +use crate::logic::{Condition, Conditions}; use crate::providers::alert_provider::enqueue_alert; use crate::providers::condition_collapse_provider::ConditionCollapseProvider; use crate::providers::editor_provider::EditorProvider; +use crate::schema::SchemaType; #[derive(Clone, Debug, Default)] pub struct Data { - pub context: Vec, + pub context: Conditions, pub overrides: Vec<(String, Value)>, } @@ -45,7 +45,7 @@ enum FormMode { #[component] fn form( - context: Vec, + context: Conditions, overrides: Vec<(String, Value)>, dimensions: Vec, edit: bool, @@ -63,13 +63,11 @@ fn form( spawn_local(async move { let f_context = context.get(); let f_overrides = overrides.get(); - let dimensions = dimensions.get_value().clone(); let result = if edit { update_context( tenant_rs.get().clone(), Map::from_iter(f_overrides), f_context, - dimensions.clone(), ) .await } else { @@ -77,7 +75,6 @@ fn form( tenant_rs.get().clone(), Map::from_iter(f_overrides), f_context, - dimensions.clone(), ) .await }; @@ -98,7 +95,6 @@ fn form( impl IntoView { let tenant_rs = use_context::>().unwrap(); - let (selected_data, set_selected_data) = create_signal::>(None); + let (selected_context_rs, selected_context_ws) = create_signal::>(None); let (form_mode, set_form_mode) = create_signal::>(None); let (modal_visible, set_modal_visible) = create_signal(false); let (delete_id, set_delete_id) = create_signal::>(None); @@ -168,71 +163,81 @@ pub fn context_override() -> impl IntoView { }, ); - let handle_context_create = Callback::new(move |_| { + let on_create_context_click = Callback::new(move |_| { set_form_mode.set(Some(FormMode::Create)); let PageResource { dimensions, .. } = page_resource.get().unwrap_or_default(); - let context_with_mandatory_dimensions = dimensions - .into_iter() - .filter_map(|dim| { - if dim.mandatory { - Some(Condition { - left_operand: dim.dimension, - operator: ConditionOperator::Is, - right_operand: vec![Value::from("")], - }) - } else { - None - } - }) - .collect::>(); - set_selected_data.set(Some(Data { - context: context_with_mandatory_dimensions, + let mut default_ctx: Conditions = Conditions(vec![]); + for dim in dimensions.iter().filter(|v| v.mandatory) { + let r#type = SchemaType::try_from(dim.schema.clone()); + if let Err(_) = r#type { + //TODO emit an alert and return + return; + } + + default_ctx.push(Condition::new_with_default_expression( + dim.dimension.clone(), + r#type.unwrap(), + )); + } + + selected_context_ws.set(Some(Data { + context: default_ctx, overrides: vec![], })); open_drawer("context_and_override_drawer"); }); - let handle_submit = Callback::new(move |_| { + let on_submit = Callback::new(move |_| { close_drawer("context_and_override_drawer"); set_form_mode.set(None); - set_selected_data.set(None); + selected_context_ws.set(None); page_resource.refetch(); }); - let handle_context_edit = - Callback::new(move |data: (Context, Map)| { - let (context, overrides) = data; - let conditions = extract_conditions(&Value::Object(context.condition.into())); - - set_selected_data.set(Some(Data { - context: conditions, - overrides: overrides.into_iter().collect::>(), - })); - set_form_mode.set(Some(FormMode::Edit)); - - open_drawer("context_and_override_drawer"); - }); + let on_context_edit = Callback::new(move |data: (Context, Map)| { + let (context, overrides) = data; + match Conditions::from_context_json(&context.condition.into()) { + Ok(conditions) => { + selected_context_ws.set(Some(Data { + context: conditions, + overrides: overrides.into_iter().collect::>(), + })); + set_form_mode.set(Some(FormMode::Edit)); + open_drawer("context_and_override_drawer"); + } + Err(e) => { + logging::error!("Error parsing context: {}", e); + enqueue_alert(e.to_string(), AlertType::Error, 5000); + } + }; + }); - let handle_context_clone = - Callback::new(move |data: (Context, Map)| { - let (context, overrides) = data; - let conditions = extract_conditions(&Value::Object(context.condition.into())); + let on_context_clone = Callback::new(move |data: (Context, Map)| { + let (context, overrides) = data; - set_selected_data.set(Some(Data { - context: conditions, - overrides: overrides.into_iter().collect::>(), - })); - set_form_mode.set(Some(FormMode::Create)); + match Conditions::from_context_json(&context.condition.into()) { + Ok(conditions) => { + selected_context_ws.set(Some(Data { + context: conditions, + overrides: overrides.into_iter().collect::>(), + })); + set_form_mode.set(Some(FormMode::Create)); - open_drawer("context_and_override_drawer"); - }); + open_drawer("context_and_override_drawer"); + } + Err(e) => { + logging::error!("Error parsing context: {}", e); + enqueue_alert(e.to_string(), AlertType::Error, 5000); + } + }; + }); - let handle_context_delete = Callback::new(move |id: String| { + let on_context_delete = Callback::new(move |id: String| { set_delete_id.set(Some(id.clone())); set_modal_visible.set(true); }); - let confirm_delete = Callback::new(move |_| { + let on_delete_confirm = Callback::new(move |_| { if let Some(id) = delete_id.get().clone() { spawn_local(async move { let result = delete_context(tenant_rs.get(), id).await; @@ -258,7 +263,7 @@ pub fn context_override() -> impl IntoView {

    Overrides

    Create Override @@ -266,7 +271,7 @@ pub fn context_override() -> impl IntoView { } + view! { } }>
    @@ -274,7 +279,7 @@ pub fn context_override() -> impl IntoView { let PageResource { config: _, dimensions, default_config } = page_resource .get() .unwrap_or_default(); - let data = selected_data.get(); + let data = selected_context_rs.get(); let drawer_header = match form_mode.get() { Some(FormMode::Edit) => "Update Overrides", Some(FormMode::Create) => "Create Overrides", @@ -287,7 +292,7 @@ pub fn context_override() -> impl IntoView { handle_close=move || { close_drawer("context_and_override_drawer"); set_form_mode.set(None); - set_selected_data.set(None); + selected_context_ws.set(None); } > @@ -300,7 +305,7 @@ pub fn context_override() -> impl IntoView { overrides=data.overrides dimensions=dimensions default_config=default_config - handle_submit=handle_submit + handle_submit=on_submit edit=true /> } @@ -314,7 +319,7 @@ pub fn context_override() -> impl IntoView { overrides=overrides dimensions=dimensions default_config=default_config - handle_submit=handle_submit + handle_submit=on_submit edit=false /> } @@ -378,9 +383,9 @@ pub fn context_override() -> impl IntoView { } }) @@ -394,7 +399,7 @@ pub fn context_override() -> impl IntoView { impl IntoView { ) }, |(current_tenant, filters, pagination_filters)| async move { - let fetch_all_filters = ListFilters { + let fetch_all_filters = PaginationParams { page: None, count: None, all: Some(true), @@ -221,7 +223,7 @@ pub fn experiment_list() -> impl IntoView { | { - let context = match row.get("context") { - Some(value) => value.to_owned(), - None => json!(""), - }; + let context = row + .get("context") + .and_then(|v| v.as_object().cloned()) + .unwrap_or(Map::new()); let id = row.get("id").map_or(String::from(""), |value| { value.as_str().unwrap_or("").to_string() }); + let conditions = + Conditions::from_context_json(&context).unwrap_or_default(); view! {
    - +
    } .into_view() diff --git a/crates/frontend/src/pages/home.rs b/crates/frontend/src/pages/home.rs index ef8890a7..7c39141b 100644 --- a/crates/frontend/src/pages/home.rs +++ b/crates/frontend/src/pages/home.rs @@ -1,4 +1,3 @@ -use std::borrow::Cow; use std::time::Duration; use leptos::*; @@ -9,12 +8,9 @@ use superposition_types::{custom_query::PaginationParams, Config}; use wasm_bindgen::JsCast; use web_sys::{HtmlButtonElement, HtmlSpanElement, MouseEvent}; -use crate::components::condition_pills::types::Conditions; -use crate::components::condition_pills::{ - types::{Condition, ConditionOperator}, - Condition as ConditionComponent, -}; +use crate::components::condition_pills::Condition as ConditionComponent; use crate::components::skeleton::{Skeleton, SkeletonVariant}; +use crate::logic::Conditions; use crate::providers::condition_collapse_provider::ConditionCollapseProvider; use crate::{ api::{fetch_config, fetch_dimensions}, @@ -64,14 +60,9 @@ fn all_context_view(config: Config) -> impl IntoView { overrides, default_configs, } = config; - let rows = |k: &String, v: &Value, striked: bool| { - let default_iter = vec![(k.clone(), v.clone())]; - v - .as_object() - .unwrap_or(&Map::from_iter(default_iter)) - .iter() + let rows = |m: &Map, striked: bool| { + m.iter() .map(|(key, value)| { - let key = key.replace('"', "").trim().to_string(); let value = value .as_str() .unwrap_or(value.to_string().trim_matches('"')) @@ -113,7 +104,7 @@ fn all_context_view(config: Config) -> impl IntoView { let rows: Vec<_> = context .override_with_keys .iter() - .filter_map(|key| overrides.get(key).map(|o| rows(key, &Value::Object(o.clone().into()), true))) + .filter_map(|key| overrides.get(key).map(|o| rows(o, true))) .collect(); let conditions: Conditions = context.try_into().unwrap_or_default(); view! { @@ -123,7 +114,7 @@ fn all_context_view(config: Config) -> impl IntoView {
    @@ -158,10 +149,7 @@ fn all_context_view(config: Config) -> impl IntoView { - {default_configs - .iter() - .map(|(k, v)| rows(k, v, false)) - .collect::>()} + {rows(&default_configs, false)} @@ -187,7 +175,7 @@ pub fn home() -> impl IntoView { }, ); - let (context_rs, context_ws) = create_signal::>(vec![]); + let (context_rs, context_ws) = create_signal::(Conditions::default()); let (selected_tab_rs, selected_tab_ws) = create_signal(ResolveTab::AllConfig); let (req_inprogess_rs, req_inprogress_ws) = create_signal(false); @@ -245,42 +233,6 @@ pub fn home() -> impl IntoView { } }; - let gen_query_context = |query: Vec| -> String { - let mut context: Vec = vec![]; - for condition in query.iter() { - let dimension = condition.left_operand.clone(); - let op = match condition.operator.clone() { - ConditionOperator::Is => Cow::Borrowed("="), - ConditionOperator::In => Cow::Borrowed("IN"), - ConditionOperator::Has => Cow::Borrowed("HAS"), - ConditionOperator::Between => Cow::Borrowed("BETWEEN"), - ConditionOperator::Other(op) => Cow::Owned(op), - }; - let value = condition - .right_operand - .clone() - .into_iter() - .filter_map(|value| { - if value.is_object() && value.get("var").is_some() { - None - } else { - Some(value) - } - }) - .map(|value| match value { - Value::String(s) => s.clone(), - Value::Number(n) => n.to_string(), - Value::Bool(b) => b.to_string(), - Value::Null => String::from("null"), - _ => format!("{}", value), - }) - .collect::>() - .join(","); - context.push(format!("{}{op}{}", dimension, value)); - } - context.join("&").to_string() - }; - let resolve_click = move |ev: MouseEvent| { ev.prevent_default(); req_inprogress_ws.set(true); @@ -308,7 +260,7 @@ pub fn home() -> impl IntoView { let context_updated = context_rs.get(); // resolve the context and get the config that would apply spawn_local(async move { - let context = gen_query_context(context_updated); + let context = context_updated.to_query_string(); let mut config = match resolve_config(tenant_rs.get_untracked(), context) .await .unwrap() @@ -389,10 +341,9 @@ pub fn home() -> impl IntoView { .to_owned() .unwrap_or_default() .data - context=vec![] + context=Conditions::default() heading_sub_text="Query your configs".to_string() dropdown_direction=DropdownDirection::Right - is_standalone=false resolve_mode=true handle_change=move |new_context| { context_ws diff --git a/crates/frontend/src/types.rs b/crates/frontend/src/types.rs index 887c9b48..d73fe6c6 100644 --- a/crates/frontend/src/types.rs +++ b/crates/frontend/src/types.rs @@ -15,11 +15,9 @@ use superposition_types::{ Exp, Overrides, SortBy, }; +use crate::logic::Conditions; use crate::{ - components::{ - condition_pills::{types::Condition, utils::extract_conditions}, - dropdown::utils::DropdownOption, - }, + components::dropdown::utils::DropdownOption, pages::experiment_list::utils::ExperimentSortOn, }; @@ -194,7 +192,7 @@ pub struct Experiment { pub(crate) name: String, pub(crate) id: String, pub(crate) traffic_percentage: u8, - pub(crate) context: Vec, + pub(crate) context: Conditions, pub(crate) status: ExperimentStatusType, pub(crate) override_keys: Value, pub(crate) created_by: String, @@ -216,7 +214,10 @@ impl From for Experiment { last_modified: value.last_modified, chosen_variant: value.chosen_variant, variants: serde_json::from_value(value.variants).unwrap_or_default(), - context: extract_conditions(&value.context), + context: Conditions::from_context_json( + value.context.as_object().unwrap_or(&Map::new()), + ) + .unwrap_or(Conditions(vec![])), } } } diff --git a/crates/frontend/src/utils.rs b/crates/frontend/src/utils.rs index 6d23610f..97bbf15e 100644 --- a/crates/frontend/src/utils.rs +++ b/crates/frontend/src/utils.rs @@ -4,10 +4,6 @@ use cfg_if::cfg_if; use leptos::*; use reqwest::header::{HeaderMap, HeaderName, HeaderValue}; use serde::de::DeserializeOwned; -use serde_json::{Map, Value}; -use superposition_types::database::{ - models::cac::DefaultConfig, types::DimensionWithMandatory, -}; use url::Url; use wasm_bindgen::JsCast; @@ -197,170 +193,6 @@ pub fn check_url_and_return_val(s: String) -> String { } } -pub enum ConfigType { - // Default config is not actually dead code, it's used - // but for some reason the compiler thinks it's dead code - // this gets rid of the warning - #[allow(dead_code)] - DefaultConfig(DefaultConfig), - Dimension(DimensionWithMandatory), -} - -#[derive(Clone, Debug)] -pub enum ConfigValueType { - Boolean, - Number, - String, - Null, - Integer, - Other, -} - -impl FromStr for ConfigValueType { - type Err = String; - fn from_str(s: &str) -> Result { - match s { - "boolean" => Ok(ConfigValueType::Boolean), - "number" => Ok(ConfigValueType::Number), - "string" => Ok(ConfigValueType::String), - "null" => Ok(ConfigValueType::Null), - "integer" => Ok(ConfigValueType::Integer), - _ => Ok(ConfigValueType::Other), - } - } -} - -pub fn get_config_type( - configs: &[ConfigType], - key_name: &str, -) -> Option> { - let config = configs.iter().find(|conf| match conf { - ConfigType::DefaultConfig(default_conf) => default_conf.key == key_name, - ConfigType::Dimension(dimension) => dimension.dimension == key_name, - }); - - let type_from_str = |type_str: Option<&str>| { - type_str - .map(|t| ConfigValueType::from_str(t).ok()) - .flatten() - .unwrap_or(ConfigValueType::Other) - }; - - let types_mapping = |schema_type: &Value| match schema_type { - Value::Array(types) => types - .iter() - .map(|item: &Value| type_from_str(item.as_str())) - .collect::>(), - Value::String(type_str) => vec![type_from_str(Some(type_str.as_str()))], - _ => vec![ConfigValueType::Other], - }; - - config.and_then(|config| match config { - ConfigType::DefaultConfig(default_conf) => { - default_conf.schema.get("type").map(|t| types_mapping(t)) - } - - ConfigType::Dimension(dimension) => { - dimension.schema.get("type").map(|t| types_mapping(t)) - } - }) -} - -pub fn parse_value(val: &Value, config_type: ConfigValueType) -> Result { - match config_type { - ConfigValueType::Boolean => { - match val { - Value::Bool(_) => Ok(val.clone()), - - Value::String(s) => { - // Attempting to parse the string as a boolean - match s.to_lowercase().as_str() { - "true" => Ok(Value::Bool(true)), - "false" => Ok(Value::Bool(false)), - _ => Err(format!("Invalid boolean string: {:?}", s)), // Error if not a valid boolean string - } - } - - _ => Err(format!("Invalid boolean value: {:?}", val)), - } - } - - ConfigValueType::Number | ConfigValueType::Integer => { - match val { - Value::Number(num) => Ok(Value::Number(num.clone())), - - Value::String(s) => { - // Attempting to parse as integer first, then as float - if let Ok(int_val) = s.parse::() { - Ok(Value::Number(int_val.into())) - } else if let Ok(float_val) = s.parse::() { - Ok(Value::Number( - serde_json::Number::from_f64(float_val).unwrap(), - )) - } else { - Err(format!("Invalid number format: {:?}", s)) - } - } - - Value::Array(arr) => { - // Ensuring all items in the array are numbers - if arr.iter().all(|item| item.is_number()) { - Ok(val.clone()) - } else { - Err("Array contains non-number value".to_string()) - } - } - - _ => Err(format!( - "{:?} is neither a valid number nor an array of numbers.", - val - )), - } - } - - ConfigValueType::String => match val { - Value::String(_) => Ok(val.clone()), - Value::Number(i) => Ok(Value::String(i.to_string())), - Value::Bool(b) => Ok(Value::String(b.to_string())), - Value::Array(arr) => { - if arr.iter().all(|item| item.is_string()) { - Ok(val.clone()) - } else { - Err("Array contains non-string value".to_string()) - } - } - _ => Err(format!( - "{:?} is neither a valid string nor an array of strings.", - val - )), - }, - - ConfigValueType::Null if val.is_null() => Ok(Value::Null), - - _ => Ok(val.clone()), - } -} - -pub fn get_config_value( - name: &str, - val: &Value, - configs: &[ConfigType], -) -> Result { - let config_type = get_config_type(configs, name); - - match config_type { - Some(possible_types) => { - for possible_type in possible_types { - if let Ok(parsed_value) = parse_value(val, possible_type) { - return Ok(parsed_value); - } - } - Err("Error parsing config value".to_string()) - } - None => Ok(val.clone()), - } -} - /********* Request Utils **********/ use once_cell::sync::Lazy; @@ -489,17 +321,6 @@ pub fn set_local_storage(_key: &str, _value: &str) -> Option<()> { } } -pub fn get_key_type(schema: &Map) -> String { - if schema.contains_key("enum") { - String::from("ENUM") - } else { - match schema.get("type").unwrap_or(&Value::Null) { - Value::String(str_) => str_.to_ascii_uppercase(), - _ => String::from("STRING"), - } - } -} - pub fn update_page_direction( page: Option, total_pages: i64,