remix-client-cache
remix-client-cache is a powerful and lightweight library made for Remix.run to cache your server loader data on the client using clientLoaders.
By default it uses the stale while revalidate strategy and hot swaps your stale info once loaded from the server. It also allows you to invalidate the cache for a specific key or multiple keys.
It allows you to pass in an adapter of your choice to cache your data.
It comes with a default adapter that uses in memory storage to cache your data.
First party support for localStorage, sessionStorage and localforage packages. You can just provide them as the argument to configureGlobalCache
.
Install
npm install remix-client-cache
Basic usage
Here is an example usage of remix-client-cache with the default in memory adapter.
import { json, type LoaderFunctionArgs } from "@remix-run/node";
import { ClientLoaderFunctionArgs } from "@remix-run/react";
import { cacheClientLoader, useCachedLoaderData } from "remix-client-cache";
export const loader = async ({ params }: LoaderFunctionArgs) => {
const response = await fetch(
`https://jsonplaceholder.typicode.com/users/${params.user}`
);
const user = await response.json();
await new Promise((resolve) => setTimeout(resolve, 1000));
return json({ user: { ...user, description: Math.random() } });
};
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader(args);
clientLoader.hydrate = true;
export default function Index() {
const { user } = useCachedLoaderData<typeof loader>();
return (
<div>
{user.name} <hr /> {user.email}
<hr />
{user.username}
<hr />
{user.website} <hr />
{user.description}
</div>
);
}
Cache adapters
The library exports an interface you need to implement in order to create your own cache adapter. The interface is called CacheAdapter
.
It closely matches the interface of Storage
and requires you to have the following methods:
getItem
: takes a key and returns a promise that resolves to the value stored at that keysetItem
: takes a key and a value and returns a promise that resolves when the value is storedremoveItem
: takes a key and returns a promise that resolves when the value is removed
The cacheLoaderData
will use the default memory cache adapter that comes with the library. If you want an advanced use-case make sure that the adapter you provide implements the CacheAdapter
interface.
import { RemixBrowser } from "@remix-run/react";
import { startTransition, StrictMode } from "react";
import { hydrateRoot } from "react-dom/client";
import { configureGlobalCache } from "remix-client-cache";
configureGlobalCache(() => localStorage);
startTransition(() => {
hydrateRoot(
document,
<StrictMode>
<RemixBrowser />
</StrictMode>
);
});
You can use the configureGlobalCache
function to override the libraries default in-memory cache adapter. It will globally switch to whatever adapter you provide to it.
If you want to have a per route adapter you can use the createCacheAdapter
to create an adapter and provide it to your hooks and functions.
import { createCacheAdapter, useCachedLoaderData } from "remix-client-cache";
const { adapter } = createCacheAdapter(() => localStorage);
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader(args, {
adapter
});
clientLoader.hydrate = true;
export default function Index() {
const { user } = useCachedLoaderData<typeof loader>({
adapter
});
return (
<div>
{user.name} <hr /> {user.email}
<hr />
{user.username}
<hr />
{user.website} <hr />
{user.description}
</div>
);
}
Here are some examples of how you can use the library with different global adapters.
configureGlobalCache(() => localStorage);
configureGlobalCache(() => sessionStorage);
configureGlobalCache(() => localforage);
Also with different per route adapters:
const { adapter } = createCacheAdapter(() => localStorage);
const { adapter } = createCacheAdapter(() => sessionStorage);
const { adapter } = createCacheAdapter(() => localforage);
Let's say you want to use a custom adapter that uses a database to store the data.
You can do that by implementing the CacheAdapter
interface and passing it to the configureGlobalCache
or createCacheAdapter
function.
class DatabaseAdapter implements CacheAdapter {
async getItem(key: string) {
}
async setItem(key: string, value: string) {
}
async removeItem(key: string) {
}
}
configureGlobalCache(() => new DatabaseAdapter());
const { adapter } = createCacheAdapter(() => new DatabaseAdapter());
API's
createCacheAdapter
Function that creates a cache adapter and returns it. It takes one argument, the adapter
that is used to store the data.
import { createCacheAdapter } from "remix-client-cache";
const { adapter } = createCacheAdapter(() => localStorage);
configureGlobalCache
Function that configures the global cache adapter. It takes one argument, the adapter
that is used to store the data.
import { configureGlobalCache } from "remix-client-cache";
configureGlobalCache(() => localStorage);
cacheClientLoader
Used to cache the data that is piped from the loader to your component using the clientLoader
export.
It takes two arguments, the first one is the ClientLoaderFunctionArgs
object that is passed to the clientLoader
function, the second one is an object with the following properties:
type
- that tells the client loader if it should use the normal caching mechanism where it stores the data and early returns that instead of refetching or if it should use the staleWhileRevalidate
mechanism where it returns the cached data and refetches in the background.key
- key that is used to store the data in the cache. Defaults to the current route path including search params and hashes. (eg. /user/1?name=John#profile)adapter
- the cache adapter that is used to store the data. Defaults to the in memory adapter that comes with the library.
import { json, type LoaderFunctionArgs } from "@remix-run/node";
import { ClientLoaderFunctionArgs } from "@remix-run/react";
import { cacheClientLoader, useCachedLoaderData } from "remix-client-cache";
export const loader = async ({ params }: LoaderFunctionArgs) => {
const response = await fetch(
`https://jsonplaceholder.typicode.com/users/${params.user}`
);
const user = await response.json();
await new Promise((resolve) => setTimeout(resolve, 1000));
return json({ user: { ...user, description: Math.random() } });
};
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader(args, {
type: "swr",
key: "/user/1"
adapter: () => localStorage
});
clientLoader.hydrate = true;
decacheClientLoader
Used to remove the data that is piped from the loader to your component using the clientLoader
export.
import { json, type LoaderFunctionArgs } from "@remix-run/node";
import { ClientLoaderFunctionArgs } from "@remix-run/react";
import { decacheClientLoader, useCachedLoaderData } from "remix-client-cache";
export const loader = async ({ params }: LoaderFunctionArgs) => {
const response = await fetch(
`https://jsonplaceholder.typicode.com/users/${params.user}`
);
const user = await response.json();
await new Promise((resolve) => setTimeout(resolve, 1000));
return json({ user: { ...user, description: Math.random() } });
};
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader;
clientLoader.hydrate = true;
export const clientAction = decacheClientLoader;
Accepts an optional object with the following properties:
key
- key that is used to store the data in the cache.adapter
- the cache adapter that is used to store the data.
useCachedLoaderData
Hook that can be used to get the cached data from the clientLoader
export. Must be used together with cacheClientLoader
because the data returned from
the cacheClientLoader
is augmented to work with useCachedLoaderData
in mind and not the standard useLoaderData
hook.
import { useCachedLoaderData } from "remix-client-cache";
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader(args, "swr");
clientLoader.hydrate = true;
export default function Index() {
const { user } = useCachedLoaderData<typeof loader>();
return (
<div>
{user.name} <hr /> {user.email}
<hr />
{user.username}
<hr />
{user.website} <hr />
{user.description}
</div>
);
}
Accepts an optional object with the following properties:
useSwrData
Hook used to get an SWR component that hot swaps the data for you. It takes one argument, loaderData returned by the useCachedLoaderData
OR useLoaderData
hook.
import { useCachedLoaderData, useSwrData } from "remix-client-cache";
export const clientLoader = (args: ClientLoaderFunctionArgs) => cacheClientLoader(args);
clientLoader.hydrate = true;
export default function Index() {
const loaderData = useLoaderData<typeof loader>();
const loaderData = useCachedLoaderData<typeof loader>();
const SWR = useSwrData(loaderData);
return (
<SWR>
{/** Hot swapped automatically */}
{({ user }) => (
<div>
{data.name} <hr /> {data.email}
<hr />
{data.username}
<hr />
{data.website} <hr />
{data.description}
</div>
)}
</SWR>
);
}
invalidateCache
Utility function that can be used to invalidate the cache for a specific key. It takes one argument, the key
that is used to store the data in the cache.
Can also be an array of keys
import { invalidateCache } from "remix-client-cache";
invalidateCache("/user/1");
Keep in mind this can only be used on the client, so either in clientLoader
or clientAction
exports or the components themselves.
useCacheInvalidator
Hook that returns a function that can be used to invalidate the cache for a specific key. It takes one argument, the key
that is used to store the data in the cache. Can also be an array of keys
import { useCacheInvalidator } from "remix-client-cache";
export default function Index() {
const { invalidateCache } = useCacheInvalidator();
return (
<div>
// invalidates the cache for the /user/1 route
<button onClick={ () => invalidateCache("/user/1") }>Invalidate cache</button>
</div>
);
}
Support
If you like the project, please consider supporting us by giving a ⭐️ on Github.
License
MIT
Bugs
If you find a bug, please file an issue on our issue tracker on GitHub
Contributing
Thank you for considering contributing to remix-client-cache! We welcome any contributions, big or small, including bug reports, feature requests, documentation improvements, or code changes.
To get started, please fork this repository and make your changes in a new branch. Once you're ready to submit your changes, please open a pull request with a clear description of your changes and any related issues or pull requests.
Please note that all contributions are subject to our Code of Conduct. By participating, you are expected to uphold this code.
We appreciate your time and effort in contributing to remix-client-cache and helping to make it a better tool for the community!