123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- use crate::{
- scope_context::{consume_context, current_scope_id, schedule_update_any},
- Element, IntoDynNode, LazyNodes, Properties, Scope, ScopeId, ScopeState, Template,
- TemplateAttribute, TemplateNode, VNode,
- };
- use std::{
- any::{Any, TypeId},
- backtrace::Backtrace,
- cell::{Cell, RefCell},
- error::Error,
- fmt::{Debug, Display},
- rc::Rc,
- sync::Arc,
- };
- /// Provide an error boundary to catch errors from child components
- pub fn use_error_boundary(cx: &ScopeState) -> &ErrorBoundary {
- cx.use_hook(|| cx.provide_context(ErrorBoundary::new()))
- }
- /// A boundary that will capture any errors from child components
- #[derive(Debug, Clone, Default)]
- pub struct ErrorBoundary {
- inner: Rc<ErrorBoundaryInner>,
- }
- /// A boundary that will capture any errors from child components
- pub struct ErrorBoundaryInner {
- error: RefCell<Option<CapturedError>>,
- _id: ScopeId,
- rerun_boundary: Arc<dyn Fn(ScopeId) + Send + Sync>,
- }
- impl Debug for ErrorBoundaryInner {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- f.debug_struct("ErrorBoundaryInner")
- .field("error", &self.error)
- .finish()
- }
- }
- #[derive(Debug)]
- /// An instance of an error captured by a descendant component.
- pub struct CapturedError {
- /// The error captured by the error boundary
- pub error: Box<dyn Debug + 'static>,
- /// The backtrace of the error
- pub backtrace: Backtrace,
- /// The scope that threw the error
- pub scope: ScopeId,
- }
- impl Display for CapturedError {
- fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
- f.write_fmt(format_args!(
- "Encountered error: {:?}\nIn scope: {:?}\nBacktrace: {}",
- self.error, self.scope, self.backtrace
- ))
- }
- }
- impl Error for CapturedError {}
- impl CapturedError {
- /// Downcast the error type into a concrete error type
- pub fn downcast<T: 'static>(&self) -> Option<&T> {
- if TypeId::of::<T>() == self.error.type_id() {
- let raw = self.error.as_ref() as *const _ as *const T;
- Some(unsafe { &*raw })
- } else {
- None
- }
- }
- }
- impl Default for ErrorBoundaryInner {
- fn default() -> Self {
- Self {
- error: RefCell::new(None),
- _id: current_scope_id()
- .expect("Cannot create an error boundary outside of a component's scope."),
- rerun_boundary: schedule_update_any().unwrap(),
- }
- }
- }
- impl ErrorBoundary {
- /// Create a new error boundary
- pub fn new() -> Self {
- Self::default()
- }
- /// Create a new error boundary in the current scope
- pub(crate) fn new_in_scope(
- scope: ScopeId,
- rerun_boundary: Arc<dyn Fn(ScopeId) + Send + Sync>,
- ) -> Self {
- Self {
- inner: Rc::new(ErrorBoundaryInner {
- error: RefCell::new(None),
- _id: scope,
- rerun_boundary,
- }),
- }
- }
- /// Push an error into this Error Boundary
- pub fn insert_error(
- &self,
- scope: ScopeId,
- error: Box<dyn Debug + 'static>,
- backtrace: Backtrace,
- ) {
- println!("{:?} {:?}", error, self.inner._id);
- self.inner.error.replace(Some(CapturedError {
- error,
- scope,
- backtrace,
- }));
- (self.inner.rerun_boundary)(self.inner._id);
- }
- /// Take any error that has been captured by this error boundary
- pub fn take_error(&self) -> Option<CapturedError> {
- self.inner.error.take()
- }
- }
- /// A trait to allow results to be thrown upwards to the nearest Error Boundary
- ///
- /// The canonical way of using this trait is to throw results from hooks, aborting rendering
- /// through question mark syntax. The throw method returns an option that evaluates to None
- /// if there is an error, injecting the error to the nearest error boundary.
- ///
- /// If the value is `Ok`, then throw returns the value, not aborting the rendering process.
- ///
- /// The call stack is saved for this component and provided to the error boundary
- ///
- /// ```rust, ignore
- /// #[component]
- /// fn App(cx: Scope, count: String) -> Element {
- /// let id: i32 = count.parse().throw()?;
- ///
- /// cx.render(rsx! {
- /// div { "Count {}" }
- /// })
- /// }
- /// ```
- pub trait Throw<S = ()>: Sized {
- /// The value that will be returned in if the given value is `Ok`.
- type Out;
- /// Returns an option that evaluates to None if there is an error, injecting the error to the nearest error boundary.
- ///
- /// If the value is `Ok`, then throw returns the value, not aborting the rendering process.
- ///
- /// The call stack is saved for this component and provided to the error boundary
- ///
- ///
- /// Note that you can also manually throw errors using the throw method on `ScopeState` directly,
- /// which is what this trait shells out to.
- ///
- ///
- /// ```rust, ignore
- /// #[component]
- /// fn App(cx: Scope, count: String) -> Element {
- /// let id: i32 = count.parse().throw()?;
- ///
- /// cx.render(rsx! {
- /// div { "Count {}" }
- /// })
- /// }
- /// ```
- fn throw(self) -> Option<Self::Out>;
- /// Returns an option that evaluates to None if there is an error, injecting the error to the nearest error boundary.
- ///
- /// If the value is `Ok`, then throw returns the value, not aborting the rendering process.
- ///
- /// The call stack is saved for this component and provided to the error boundary
- ///
- ///
- /// Note that you can also manually throw errors using the throw method on `ScopeState` directly,
- /// which is what this trait shells out to.
- ///
- ///
- /// ```rust, ignore
- /// #[component]
- /// fn App(cx: Scope, count: String) -> Element {
- /// let id: i32 = count.parse().throw()?;
- ///
- /// cx.render(rsx! {
- /// div { "Count {}" }
- /// })
- /// }
- /// ```
- fn throw_with<D: Debug + 'static>(self, e: impl FnOnce() -> D) -> Option<Self::Out> {
- self.throw().or_else(|| throw_error(e()))
- }
- }
- fn throw_error<T>(e: impl Debug + 'static) -> Option<T> {
- if let Some(cx) = consume_context::<ErrorBoundary>() {
- match current_scope_id() {
- Some(id) => cx.insert_error(id, Box::new(e), Backtrace::capture()),
- None => {
- tracing::error!("Cannot throw error outside of a component's scope.")
- }
- }
- }
- None
- }
- /// We call clone on any errors that can be owned out of a reference
- impl<'a, T, O: Debug + 'static, E: ToOwned<Owned = O>> Throw for &'a Result<T, E> {
- type Out = &'a T;
- fn throw(self) -> Option<Self::Out> {
- match self {
- Ok(t) => Some(t),
- Err(e) => throw_error(e.to_owned()),
- }
- }
- fn throw_with<D: Debug + 'static>(self, err: impl FnOnce() -> D) -> Option<Self::Out> {
- match self {
- Ok(t) => Some(t),
- Err(_e) => throw_error(err()),
- }
- }
- }
- /// Or just throw errors we know about
- impl<T, E: Debug + 'static> Throw for Result<T, E> {
- type Out = T;
- fn throw(self) -> Option<T> {
- match self {
- Ok(t) => Some(t),
- Err(e) => throw_error(e),
- }
- }
- fn throw_with<D: Debug + 'static>(self, error: impl FnOnce() -> D) -> Option<Self::Out> {
- self.ok().or_else(|| throw_error(error()))
- }
- }
- /// Or just throw errors we know about
- impl<T> Throw for Option<T> {
- type Out = T;
- fn throw(self) -> Option<T> {
- self.or_else(|| throw_error("Attempted to unwrap a None value."))
- }
- fn throw_with<D: Debug + 'static>(self, error: impl FnOnce() -> D) -> Option<Self::Out> {
- self.or_else(|| throw_error(error()))
- }
- }
- pub struct ErrorHandler<'a>(Box<dyn Fn(CapturedError) -> LazyNodes<'a, 'a> + 'a>);
- impl<'a, F: Fn(CapturedError) -> LazyNodes<'a, 'a> + 'a> From<F> for ErrorHandler<'a> {
- fn from(value: F) -> Self {
- Self(Box::new(value))
- }
- }
- fn default_handler<'a>(error: CapturedError) -> LazyNodes<'a, 'a> {
- LazyNodes::new(move |__cx: &ScopeState| -> VNode {
- static TEMPLATE: Template = Template {
- name: "error_handle.rs:42:5:884",
- roots: &[TemplateNode::Element {
- tag: "pre",
- namespace: None,
- attrs: &[TemplateAttribute::Static {
- name: "color",
- namespace: Some("style"),
- value: "red",
- }],
- children: &[TemplateNode::DynamicText { id: 0usize }],
- }],
- node_paths: &[&[0u8, 0u8]],
- attr_paths: &[],
- };
- VNode {
- parent: Default::default(),
- stable_id: Default::default(),
- key: None,
- template: std::cell::Cell::new(TEMPLATE),
- root_ids: bumpalo::collections::Vec::with_capacity_in(1usize, __cx.bump()).into(),
- dynamic_nodes: __cx
- .bump()
- .alloc([__cx.text_node(format_args!("{0}", error))]),
- dynamic_attrs: __cx.bump().alloc([]),
- }
- })
- }
- pub struct ErrorBoundaryProps<'a> {
- children: Element<'a>,
- handle_error: ErrorHandler<'a>,
- }
- impl<'a> ErrorBoundaryProps<'a> {
- /**
- Create a builder for building `ErrorBoundaryProps`.
- On the builder, call `.children(...)`(optional), `.handle_error(...)`(optional) to set the values of the fields.
- Finally, call `.build()` to create the instance of `ErrorBoundaryProps`.
- */
- #[allow(dead_code)]
- pub fn builder() -> ErrorBoundaryPropsBuilder<'a, ((), ())> {
- ErrorBoundaryPropsBuilder {
- fields: ((), ()),
- _phantom: ::core::default::Default::default(),
- }
- }
- }
- #[must_use]
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, non_snake_case)]
- pub struct ErrorBoundaryPropsBuilder<'a, TypedBuilderFields> {
- fields: TypedBuilderFields,
- _phantom: ::core::marker::PhantomData<&'a ()>,
- }
- impl<'a, TypedBuilderFields> Clone for ErrorBoundaryPropsBuilder<'a, TypedBuilderFields>
- where
- TypedBuilderFields: Clone,
- {
- fn clone(&self) -> Self {
- Self {
- fields: self.fields.clone(),
- _phantom: ::core::default::Default::default(),
- }
- }
- }
- impl<'a> Properties<'a> for ErrorBoundaryProps<'a> {
- type Builder = ErrorBoundaryPropsBuilder<'a, ((), ())>;
- const IS_STATIC: bool = false;
- fn builder(_: &'a ScopeState) -> Self::Builder {
- ErrorBoundaryProps::builder()
- }
- unsafe fn memoize(&self, _: &Self) -> bool {
- false
- }
- }
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, non_snake_case)]
- pub trait ErrorBoundaryPropsBuilder_Optional<T> {
- fn into_value<F: FnOnce() -> T>(self, default: F) -> T;
- }
- impl<T> ErrorBoundaryPropsBuilder_Optional<T> for () {
- fn into_value<F: FnOnce() -> T>(self, default: F) -> T {
- default()
- }
- }
- impl<T> ErrorBoundaryPropsBuilder_Optional<T> for (T,) {
- fn into_value<F: FnOnce() -> T>(self, _: F) -> T {
- self.0
- }
- }
- #[allow(dead_code, non_camel_case_types, missing_docs)]
- impl<'a, __handle_error> ErrorBoundaryPropsBuilder<'a, ((), __handle_error)> {
- pub fn children(
- self,
- children: Element<'a>,
- ) -> ErrorBoundaryPropsBuilder<'a, ((Element<'a>,), __handle_error)> {
- let children = (children,);
- let (_, handle_error) = self.fields;
- ErrorBoundaryPropsBuilder {
- fields: (children, handle_error),
- _phantom: self._phantom,
- }
- }
- }
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, non_snake_case)]
- pub enum ErrorBoundaryPropsBuilder_Error_Repeated_field_children {}
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, missing_docs)]
- impl<'a, __handle_error> ErrorBoundaryPropsBuilder<'a, ((Element<'a>,), __handle_error)> {
- #[deprecated(note = "Repeated field children")]
- pub fn children(
- self,
- _: ErrorBoundaryPropsBuilder_Error_Repeated_field_children,
- ) -> ErrorBoundaryPropsBuilder<'a, ((Element<'a>,), __handle_error)> {
- self
- }
- }
- #[allow(dead_code, non_camel_case_types, missing_docs)]
- impl<'a, __children> ErrorBoundaryPropsBuilder<'a, (__children, ())> {
- pub fn handle_error(
- self,
- handle_error: impl ::core::convert::Into<ErrorHandler<'a>>,
- ) -> ErrorBoundaryPropsBuilder<'a, (__children, (ErrorHandler<'a>,))> {
- let handle_error = (handle_error.into(),);
- let (children, _) = self.fields;
- ErrorBoundaryPropsBuilder {
- fields: (children, handle_error),
- _phantom: self._phantom,
- }
- }
- }
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, non_snake_case)]
- pub enum ErrorBoundaryPropsBuilder_Error_Repeated_field_handle_error {}
- #[doc(hidden)]
- #[allow(dead_code, non_camel_case_types, missing_docs)]
- impl<'a, __children> ErrorBoundaryPropsBuilder<'a, (__children, (ErrorHandler<'a>,))> {
- #[deprecated(note = "Repeated field handle_error")]
- pub fn handle_error(
- self,
- _: ErrorBoundaryPropsBuilder_Error_Repeated_field_handle_error,
- ) -> ErrorBoundaryPropsBuilder<'a, (__children, (ErrorHandler<'a>,))> {
- self
- }
- }
- #[allow(dead_code, non_camel_case_types, missing_docs)]
- impl<
- 'a,
- __handle_error: ErrorBoundaryPropsBuilder_Optional<ErrorHandler<'a>>,
- __children: ErrorBoundaryPropsBuilder_Optional<Element<'a>>,
- > ErrorBoundaryPropsBuilder<'a, (__children, __handle_error)>
- {
- pub fn build(self) -> ErrorBoundaryProps<'a> {
- let (children, handle_error) = self.fields;
- let children = ErrorBoundaryPropsBuilder_Optional::into_value(children, || {
- ::core::default::Default::default()
- });
- let handle_error = ErrorBoundaryPropsBuilder_Optional::into_value(handle_error, || {
- ErrorHandler(Box::new(default_handler))
- });
- ErrorBoundaryProps {
- children,
- handle_error,
- }
- }
- }
- /// Create a new error boundary component.
- ///
- /// ## Details
- ///
- /// Error boundaries handle errors within a specific part of your application. Any errors passed in a child with [`Throw`] will be caught by the nearest error boundary.
- ///
- /// ## Example
- ///
- /// ```rust, ignore
- /// rsx!{
- /// ErrorBoundary {
- /// handle_error: |error| rsx! { "Oops, we encountered an error. Please report {error} to the developer of this application" }
- /// ThrowsError {}
- /// }
- /// }
- /// ```
- ///
- /// ## Usage
- ///
- /// Error boundaries are an easy way to handle errors in your application.
- /// They are similar to `try/catch` in JavaScript, but they only catch errors in the tree below them.
- /// Error boundaries are quick to implement, but it can be useful to individually handle errors in your components to provide a better user experience when you know that an error is likely to occur.
- #[allow(non_upper_case_globals, non_snake_case)]
- pub fn ErrorBoundary<'a>(cx: Scope<'a, ErrorBoundaryProps<'a>>) -> Element {
- let error_boundary = use_error_boundary(cx);
- match error_boundary.take_error() {
- Some(error) => cx.render((cx.props.handle_error.0)(error)),
- None => Some({
- let __cx = cx;
- static TEMPLATE: Template = Template {
- name: "examples/error_handle.rs:81:17:2342",
- roots: &[TemplateNode::Dynamic { id: 0usize }],
- node_paths: &[&[0u8]],
- attr_paths: &[],
- };
- VNode {
- parent: Cell::new(None),
- stable_id: Cell::new(None),
- key: None,
- template: std::cell::Cell::new(TEMPLATE),
- root_ids: bumpalo::collections::Vec::with_capacity_in(1usize, __cx.bump()).into(),
- dynamic_nodes: __cx.bump().alloc([{
- let ___nodes = (&cx.props.children).into_dyn_node(__cx);
- ___nodes
- }]),
- dynamic_attrs: __cx.bump().alloc([]),
- }
- }),
- }
- }
|