dioxus/packages/core/tests/miri_stress.rs

409 lines
9.5 KiB
Rust
Raw Normal View History

2022-02-11 02:00:15 +00:00
#![allow(non_snake_case)]
2021-12-19 04:03:59 +00:00
/*
Stress Miri as much as possible.
Prove that we don't leak memory and that our methods are safe.
Specifically:
- [ ] VirtualDom drops memory safely
- [ ] Borrowed components don't expose invalid pointers
- [ ] Async isn't busted
*/
use dioxus::prelude::*;
2021-12-19 04:03:59 +00:00
/// This test ensures that if a component aborts early, it is replaced with a placeholder.
/// In debug, this should also toss a warning.
#[test]
#[ignore]
2021-12-19 04:03:59 +00:00
fn test_memory_leak() {
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
let val = cx.use_hook(|| 0);
2021-12-19 04:03:59 +00:00
*val += 1;
if *val == 2 || *val == 4 {
2022-11-27 07:06:04 +00:00
return cx.render(rsx!(()));
2021-12-19 04:03:59 +00:00
}
let name = cx.use_hook(|| String::from("asd"));
2021-12-19 04:03:59 +00:00
cx.render(rsx!(
div { "Hello, world!" }
Child {}
Child {}
Child {}
Child {}
Child {}
Child {}
BorrowedChild { na: name }
BorrowedChild { na: name }
BorrowedChild { na: name }
BorrowedChild { na: name }
BorrowedChild { na: name }
2021-12-19 04:03:59 +00:00
))
}
#[derive(Props)]
struct BorrowedProps<'a> {
na: &'a str,
}
fn BorrowedChild<'a>(cx: Scope<'a, BorrowedProps<'a>>) -> Element {
render!(div {
2021-12-19 04:03:59 +00:00
"goodbye {cx.props.na}"
Child {}
Child {}
2021-12-19 04:03:59 +00:00
})
}
fn Child(cx: Scope) -> Element {
render!(div { "goodbye world" })
2021-12-21 03:33:13 +00:00
}
2022-11-27 07:06:04 +00:00
let mut dom = VirtualDom::new(app);
2021-12-19 04:03:59 +00:00
dom.rebuild();
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
}
#[test]
fn memo_works_properly() {
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
let val = cx.use_hook(|| 0);
2021-12-19 04:03:59 +00:00
*val += 1;
if *val == 2 || *val == 4 {
return None;
}
let name = cx.use_hook(|| String::from("asd"));
2021-12-19 04:03:59 +00:00
cx.render(rsx!(
2022-01-03 06:12:39 +00:00
div { "Hello, world! {name}" }
Child { na: "asdfg".to_string() }
2021-12-19 04:03:59 +00:00
))
}
#[derive(PartialEq, Props)]
struct ChildProps {
na: String,
}
fn Child(cx: Scope<ChildProps>) -> Element {
render!(div { "goodbye world" })
2021-12-19 04:03:59 +00:00
}
let mut dom = new_dom(app, ());
dom.rebuild();
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
2021-12-21 03:33:13 +00:00
dom.hard_diff(ScopeId(0));
dom.hard_diff(ScopeId(0));
}
#[test]
fn free_works_on_root_props() {
fn app(cx: Scope<Custom>) -> Element {
cx.render(rsx! {
Child { a: "alpha"}
Child { a: "beta"}
Child { a: "gamma"}
Child { a: "delta"}
2021-12-21 03:33:13 +00:00
})
}
#[derive(Props, PartialEq)]
struct ChildProps {
a: &'static str,
}
fn Child(cx: Scope<ChildProps>) -> Element {
render!("child {cx.props.a}")
2021-12-21 03:33:13 +00:00
}
struct Custom {
val: String,
}
impl Drop for Custom {
fn drop(&mut self) {
2022-01-03 06:12:39 +00:00
dbg!("dropped! {}", &self.val);
2021-12-21 03:33:13 +00:00
}
}
let mut dom = new_dom(app, Custom { val: String::from("asd") });
2021-12-21 03:33:13 +00:00
dom.rebuild();
}
#[test]
fn free_works_on_borrowed() {
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
2021-12-21 03:33:13 +00:00
cx.render(rsx! {
Child { a: "alpha", b: "asd".to_string() }
2021-12-21 03:33:13 +00:00
})
}
#[derive(Props)]
struct ChildProps<'a> {
a: &'a str,
b: String,
}
fn Child<'a>(cx: Scope<'a, ChildProps<'a>>) -> Element {
2021-12-21 03:33:13 +00:00
dbg!("rendering child");
render!("child {cx.props.a}, {cx.props.b}")
2021-12-21 03:33:13 +00:00
}
impl Drop for ChildProps<'_> {
fn drop(&mut self) {
dbg!("dropped child!");
}
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
}
#[test]
fn free_works_on_root_hooks() {
/*
On Drop, scopearena drops all the hook contents.
*/
struct Droppable<T>(T);
impl<T> Drop for Droppable<T> {
fn drop(&mut self) {
dbg!("dropping droppable");
}
}
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
let name = cx.use_hook(|| Droppable(String::from("asd")));
render!(div { "{name.0}" })
2021-12-21 03:33:13 +00:00
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
}
#[test]
fn old_props_arent_stale() {
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
2021-12-21 03:33:13 +00:00
dbg!("rendering parent");
let cnt = cx.use_hook(|| 0);
2021-12-21 03:33:13 +00:00
*cnt += 1;
if *cnt == 1 {
render!(div { Child { a: "abcdef".to_string() } })
2021-12-21 03:33:13 +00:00
} else {
render!(div { Child { a: "abcdef".to_string() } })
2021-12-21 03:33:13 +00:00
}
}
#[derive(Props, PartialEq)]
struct ChildProps {
a: String,
}
fn Child(cx: Scope<ChildProps>) -> Element {
2021-12-21 03:33:13 +00:00
dbg!("rendering child", &cx.props.a);
render!(div { "child {cx.props.a}" })
2021-12-21 03:33:13 +00:00
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dbg!("forcing update to child");
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
}
#[test]
fn basic() {
2021-12-30 02:28:28 +00:00
fn app(cx: Scope) -> Element {
render!(div {
Child { a: "abcdef".to_string() }
2021-12-21 03:33:13 +00:00
})
}
#[derive(Props, PartialEq)]
struct ChildProps {
a: String,
}
fn Child(cx: Scope<ChildProps>) -> Element {
2021-12-21 03:33:13 +00:00
dbg!("rendering child", &cx.props.a);
render!(div { "child {cx.props.a}" })
2021-12-21 03:33:13 +00:00
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
2021-12-19 04:03:59 +00:00
}
#[test]
fn leak_thru_children() {
fn app(cx: Scope) -> Element {
cx.render(rsx! {
Child {
name: "asd".to_string(),
}
});
cx.render(rsx! {
div {}
})
}
#[inline_props]
fn Child(cx: Scope, name: String) -> Element {
render!(div { "child {name}" })
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
}
2022-01-29 02:43:01 +00:00
#[test]
fn test_pass_thru() {
2022-01-30 19:08:03 +00:00
#[inline_props]
fn NavContainer<'a>(cx: Scope, children: Element<'a>) -> Element {
cx.render(rsx! {
header {
2022-02-11 02:00:15 +00:00
nav { children }
2022-01-30 19:08:03 +00:00
}
})
2022-01-29 02:43:01 +00:00
}
2022-01-30 19:08:03 +00:00
fn NavMenu(cx: Scope) -> Element {
render!( NavBrand {}
2022-01-30 19:08:03 +00:00
div {
NavStart {}
NavEnd {}
2022-01-29 02:43:01 +00:00
}
)
}
2022-01-30 19:08:03 +00:00
fn NavBrand(cx: Scope) -> Element {
render!(div {})
2022-01-30 19:08:03 +00:00
}
2022-01-29 02:43:01 +00:00
2022-01-30 19:08:03 +00:00
fn NavStart(cx: Scope) -> Element {
render!(div {})
2022-01-30 19:08:03 +00:00
}
2022-01-29 02:43:01 +00:00
2022-01-30 19:08:03 +00:00
fn NavEnd(cx: Scope) -> Element {
render!(div {})
2022-01-30 19:08:03 +00:00
}
2022-01-29 02:43:01 +00:00
2022-01-30 19:08:03 +00:00
#[inline_props]
fn MainContainer<'a>(
cx: Scope,
nav: Element<'a>,
body: Element<'a>,
footer: Element<'a>,
) -> Element {
cx.render(rsx! {
div {
class: "columns is-mobile",
div {
class: "column is-full",
2022-02-11 02:00:15 +00:00
nav,
body,
footer,
2022-01-30 19:08:03 +00:00
}
}
})
}
2022-01-29 02:43:01 +00:00
2022-01-30 19:08:03 +00:00
fn app(cx: Scope) -> Element {
let nav = cx.render(rsx! {
NavContainer {
NavMenu {}
}
});
let body = cx.render(rsx! {
div {}
});
let footer = cx.render(rsx! {
div {}
});
cx.render(rsx! {
MainContainer {
nav: nav,
body: body,
footer: footer,
}
})
}
let mut dom = new_dom(app, ());
let _ = dom.rebuild();
2022-02-11 02:00:15 +00:00
for _ in 0..40 {
2022-01-30 19:08:03 +00:00
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(0)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(1)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(2)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(2)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(2)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(3)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(3)));
dom.work_with_deadline(|| false);
dom.handle_message(SchedulerMsg::Immediate(ScopeId(3)));
dom.work_with_deadline(|| false);
}
2022-01-29 02:43:01 +00:00
}