#![cfg(not(miri))] use dioxus::prelude::Props; use dioxus_core::*; use std::{cell::Cell, collections::HashSet}; fn random_ns() -> Option<&'static str> { let namespace = rand::random::() % 2; match namespace { 0 => None, 1 => Some(Box::leak( format!("ns{}", rand::random::()).into_boxed_str(), )), _ => unreachable!(), } } fn create_random_attribute(attr_idx: &mut usize) -> TemplateAttribute<'static> { match rand::random::() % 2 { 0 => TemplateAttribute::Static { name: Box::leak(format!("attr{}", rand::random::()).into_boxed_str()), value: Box::leak(format!("value{}", rand::random::()).into_boxed_str()), namespace: random_ns(), }, 1 => TemplateAttribute::Dynamic { id: { let old_idx = *attr_idx; *attr_idx += 1; old_idx }, }, _ => unreachable!(), } } fn create_random_template_node( dynamic_node_types: &mut Vec, template_idx: &mut usize, attr_idx: &mut usize, depth: usize, ) -> TemplateNode<'static> { match rand::random::() % 4 { 0 => { let attrs = { let attrs: Vec<_> = (0..(rand::random::() % 10)) .map(|_| create_random_attribute(attr_idx)) .collect(); Box::leak(attrs.into_boxed_slice()) }; TemplateNode::Element { tag: Box::leak(format!("tag{}", rand::random::()).into_boxed_str()), namespace: random_ns(), attrs, children: { if depth > 4 { &[] } else { let children: Vec<_> = (0..(rand::random::() % 3)) .map(|_| { create_random_template_node( dynamic_node_types, template_idx, attr_idx, depth + 1, ) }) .collect(); Box::leak(children.into_boxed_slice()) } }, } } 1 => TemplateNode::Text { text: Box::leak(format!("{}", rand::random::()).into_boxed_str()), }, 2 => TemplateNode::DynamicText { id: { let old_idx = *template_idx; *template_idx += 1; dynamic_node_types.push(DynamicNodeType::Text); old_idx }, }, 3 => TemplateNode::Dynamic { id: { let old_idx = *template_idx; *template_idx += 1; dynamic_node_types.push(DynamicNodeType::Other); old_idx }, }, _ => unreachable!(), } } fn generate_paths( node: &TemplateNode<'static>, current_path: &[u8], node_paths: &mut Vec>, attr_paths: &mut Vec>, ) { match node { TemplateNode::Element { children, attrs, .. } => { for attr in *attrs { match attr { TemplateAttribute::Static { .. } => {} TemplateAttribute::Dynamic { .. } => { attr_paths.push(current_path.to_vec()); } } } for (i, child) in children.iter().enumerate() { let mut current_path = current_path.to_vec(); current_path.push(i as u8); generate_paths(child, ¤t_path, node_paths, attr_paths); } } TemplateNode::Text { .. } => {} TemplateNode::DynamicText { .. } => { node_paths.push(current_path.to_vec()); } TemplateNode::Dynamic { .. } => { node_paths.push(current_path.to_vec()); } } } enum DynamicNodeType { Text, Other, } fn create_random_template(name: &'static str) -> (Template<'static>, Vec) { let mut dynamic_node_type = Vec::new(); let mut template_idx = 0; let mut attr_idx = 0; let roots = (0..(1 + rand::random::() % 5)) .map(|_| { create_random_template_node(&mut dynamic_node_type, &mut template_idx, &mut attr_idx, 0) }) .collect::>(); assert!(!roots.is_empty()); let roots = Box::leak(roots.into_boxed_slice()); let mut node_paths = Vec::new(); let mut attr_paths = Vec::new(); for (i, root) in roots.iter().enumerate() { generate_paths(root, &[i as u8], &mut node_paths, &mut attr_paths); } let node_paths = Box::leak( node_paths .into_iter() .map(|v| &*Box::leak(v.into_boxed_slice())) .collect::>() .into_boxed_slice(), ); let attr_paths = Box::leak( attr_paths .into_iter() .map(|v| &*Box::leak(v.into_boxed_slice())) .collect::>() .into_boxed_slice(), ); ( Template { name, roots, node_paths, attr_paths }, dynamic_node_type, ) } fn create_random_dynamic_node(cx: &ScopeState, depth: usize) -> DynamicNode { let range = if depth > 5 { 1 } else { 4 }; match rand::random::() % range { 0 => DynamicNode::Placeholder(Default::default()), 1 => cx.make_node((0..(rand::random::() % 5)).map(|_| VNode { key: None, parent: Default::default(), template: Cell::new(Template { name: concat!(file!(), ":", line!(), ":", column!(), ":0"), roots: &[TemplateNode::Dynamic { id: 0 }], node_paths: &[&[0]], attr_paths: &[], }), root_ids: bumpalo::collections::Vec::new_in(cx.bump()).into(), dynamic_nodes: cx.bump().alloc([cx.component( create_random_element, DepthProps { depth, root: false }, "create_random_element", )]), dynamic_attrs: &[], })), 2 => cx.component( create_random_element, DepthProps { depth, root: false }, "create_random_element", ), 3 => { let data = String::from("borrowed data"); let bumpped = cx.bump().alloc(data); cx.component( create_random_element_borrowed, BorrowedDepthProps { borrow: &*bumpped, inner: DepthProps { depth, root: false } }, "create_random_element_borrowed", ) } _ => unreachable!(), } } fn create_random_dynamic_attr(cx: &ScopeState) -> Attribute { let value = match rand::random::() % 7 { 0 => AttributeValue::Text(Box::leak( format!("{}", rand::random::()).into_boxed_str(), )), 1 => AttributeValue::Float(rand::random()), 2 => AttributeValue::Int(rand::random()), 3 => AttributeValue::Bool(rand::random()), 4 => cx.any_value(rand::random::()), 5 => AttributeValue::None, 6 => { let value = cx.listener(|e: Event| println!("{:?}", e)); return Attribute::new("ondata", value, None, false); } _ => unreachable!(), }; Attribute::new( Box::leak(format!("attr{}", rand::random::()).into_boxed_str()), value, random_ns(), rand::random(), ) } static mut TEMPLATE_COUNT: usize = 0; #[derive(Props)] struct BorrowedDepthProps<'a> { borrow: &'a str, inner: DepthProps, } fn create_random_element_borrowed<'a>(cx: Scope<'a, BorrowedDepthProps<'a>>) -> Element<'a> { println!("{}", cx.props.borrow); let bump = cx.bump(); let allocated = bump.alloc(Scoped { scope: cx, props: &cx.props.inner }); create_random_element(allocated) } #[derive(PartialEq, Props)] struct DepthProps { depth: usize, root: bool, } fn create_random_element(cx: Scope) -> Element { if rand::random::() % 10 == 0 { cx.needs_update(); } let range = if cx.props.root { 2 } else { 3 }; let node = match rand::random::() % range { 0 | 1 => { let (template, dynamic_node_types) = create_random_template(Box::leak( format!( "{}{}", concat!(file!(), ":", line!(), ":", column!(), ":"), { unsafe { let old = TEMPLATE_COUNT; TEMPLATE_COUNT += 1; old } } ) .into_boxed_str(), )); // println!("{template:#?}"); let node = VNode { key: None, parent: None, template: Cell::new(template), root_ids: bumpalo::collections::Vec::new_in(cx.bump()).into(), dynamic_nodes: { let dynamic_nodes: Vec<_> = dynamic_node_types .iter() .map(|ty| match ty { DynamicNodeType::Text => DynamicNode::Text(VText::new(Box::leak( format!("{}", rand::random::()).into_boxed_str(), ))), DynamicNodeType::Other => { create_random_dynamic_node(cx, cx.props.depth + 1) } }) .collect(); cx.bump().alloc(dynamic_nodes) }, dynamic_attrs: cx.bump().alloc( (0..template.attr_paths.len()) .map(|_| create_random_dynamic_attr(cx)) .collect::>(), ), }; Some(node) } _ => None, }; // println!("{node:#?}"); node } // test for panics when creating random nodes and templates #[cfg(not(miri))] #[test] fn create() { for _ in 0..1000 { let mut vdom = VirtualDom::new_with_props(create_random_element, DepthProps { depth: 0, root: true }); let _ = vdom.rebuild(); } } // test for panics when diffing random nodes // This test will change the template every render which is not very realistic, but it helps stress the system #[cfg(not(miri))] #[test] fn diff() { for _ in 0..100000 { let mut vdom = VirtualDom::new_with_props(create_random_element, DepthProps { depth: 0, root: true }); let _ = vdom.rebuild(); // A list of all elements that have had event listeners // This is intentionally never cleared, so that we can test that calling event listeners that are removed doesn't cause a panic let mut event_listeners = HashSet::new(); for _ in 0..100 { for &id in &event_listeners { println!("firing event on {:?}", id); vdom.handle_event( "data", std::rc::Rc::new(String::from("hello world")), id, true, ); } { let muts = vdom.render_immediate(); for mut_ in muts.edits { if let Mutation::NewEventListener { name, id } = mut_ { println!("new event listener on {:?} for {:?}", id, name); event_listeners.insert(id); } } } } } }