Async Resources and Suspense

In any real non-trivial app, you probably need to fetch data from somewhere. This is where resources and suspense come in.

The resources API provides a simple interface for fetching and refreshing async data and suspense makes it easy to render fallbacks while the data is loading.

The Resources API

To create a new resource, call the create_isomorphic_resource function.

use sycamore::prelude::*;
use sycamore::web::create_isomorphic_resource;

struct Data {
    // Define the data format here.
}

async fn fetch_data() -> Data {
    // Perform, for instance, an HTTP request to an API endpoint.
}

let resource = create_isomorphic_resource(fetch_data);

A Resource<T> is a wrapper around a Signal<Option<T>>. The value is initially set to None while the data is loading. It is then set to Some(...) containing the value of the loaded data. This makes it convenient to display the data in your view.

view! {
    (if let Some(data) = resource.get_clone() {
        view! {
            ...
        }
    } else {
        view! {}
    })
}

Note that create_isomorphic_resource, as the name suggests, runs both on the client and on the server. If you only want data-fetching to happen on the client, you can use create_client_resource which will never load data on the server.

Right now, we do not yet have a create_server_resource function which only runs on the server because this requires some form of data-serialization and server-integration which we have not fully worked out yet.

Refreshing Resources

Resources can also have dependencies, just like memos. However, since resources are async, we cannot track reactive dependencies like we would in a synchronous context. Instead, we have to explicitly specify which dependencies the resource depends on. This can be accomplished with the on(...) utility function.

let id = create_signal(12345);
let resource = create_client_resource(on(id, move || async move {
    fetch_user(id.get()).await
}));

Under the hood, on(...) simply creates a closure that first accesses id and then constructs the future. This makes it so that we access the signal synchronously first before performing any asynchronous tasks.

Suspense

With async data, we do not want to show the UI until it is ready. This problem is solved by the Suspense component and related APIs. When a Suspense component is created, it automatically creates a new suspense boundary. Any asynchronous data accessed underneath this boundary will automatically be tracked. This includes accessing resources using the resources API.

Using Suspense lets us set a fallback view to display while we are loading the asynchronous data. For example:

view! {
    Suspense(fallback=move || view! { LoadingSpinner {} }) {
        (if let Some(data) = resource.get_clone() {
            view! {
                ...
            }
        } else {
            view! {}
        })
    }
}

Since we are accessing resource under the suspense boundary, our Suspense component will display the fallback until the resource is loaded.

Transition

Resources can also be refreshed when one of its dependencies changes. This will cause the surrounding suspense boundary to be triggered again.

This is sometimes undesired. To prevent this, just replace Suspense with Transition. This component will continue to show the old view until the new data has been loaded in, providing a smoother experience.