Security News
Research
Data Theft Repackaged: A Case Study in Malicious Wrapper Packages on npm
The Socket Research Team breaks down a malicious wrapper package that uses obfuscation to harvest credentials and exfiltrate sensitive data.
react-remote-resource
Advanced tools
Intuitive remote data management in React
react-remote-resource
simplifies using remote resources, usually api endpoints, in React applications.
react-remote-resource
creates composable resources
that act as the single point of truth for a remote resource.
A resource is meant to be used inside of React components using the useEntry
hook attached to the resource. This hook will check against an internal cache for a valid data entry. If the resource
finds a valid entry, it will return the data. Otherwise the load
function, which returns a Promise, will be invoked and thrown. The nearest RemoteResourceBoundary
(using Suspense
under the hood) will catch the Promise from the load
function and render the fallback
until all outstanding Promises resolve. If any Promise rejects, the RemoteResourceBoundary
will render the required renderError
prop and onLoadError
will be called, if provided. Otherwise children
will be rendered.
This provides a straightforward and consistent way to use data from remote resources throughout your app without over-fetching, or the headache and boilerplate of Redux or some other data management library.
npm install react-remote-resource --save
// or
yarn add react-remote-resource
import {
createSingleEntryResource,
createKeyedResource,
useEntry,
RemoteResourceBoundary,
} from "react-remote-resource";
const userResource = createSingleEntryResource(
userId => fetchJson(`/api/users/${userId}`
);
const tweetsResource = createKeyedResource(
userId => userId,
userId => fetchJson(`/api/users/${userId}/tweets`),
10000,
);
const UserInfo = ({ userId }) => {
const [user] = userResource.useEntry(userId);
const [tweets] = tweetsResource.useEntry(userId);
return (
<div>
<img src={user.imageUrl} />
<h1>
{user.name} | Tweets: {tweets.length}
</h1>
<p>{user.bio}</p>
</div>
);
};
const Tweets = ({ userId }) => {
const [tweets] = tweetsResource.useEntry(userId);
return (
<>
<button type="button" onClick={() => tweetsResource.refresh(userId)}>Refresh</button>
<ul>
{tweets.map(tweet => (
<li key={tweet.id}>
<article>
<p>{tweet.message}</p>
<footer>
<small>{tweet.date}</small>
</footer>
</article>
</li>
))}
</ul>
</>
);
};
const UserProfile = ({ userId }) => (
<RemoteResourceBoundary
fallback={<p>Loading...</p>}
renderError={({ error, retry }) => (
<div>
<p>{error}</p>
<button onClick={retry}>Retry</button>
</div>
)}
>
<UserInfo userId={userId} />
<Tweets userId={userId} />
</RemoteResourceBoundary>
);
Each resource creator will return a resource
in the following shape:
type Resource<A> = {
// A function that takes the current state and a refresh flag and returns a function that takes any arguments and returns the next state or a Promise that resolves with the next state. Note: It is up to you to handle a rejected Promise. A `RemoteResourceBoundary` will not catch it.
refresh: (...args: Array<any>) => Promise<A>,
// Returns the current state of the resource
getState: () => A,
// A function that takes the next state or a function that receives the current state and returns the next state.
setState: (A | A => A) => void,
// Allows for subscribing to resource state changes. Basically a wrapper around store.subscribe.
subscribe: (() => void) => void,
// A react hook that allows you to use a resource entry's state in the same way that you would use React's useState
useEntry: <A>(...args: Array<any>) => [A, (A | A => A) => void]
};
createResource
Creates a new resource.
const productsResource = createResource(
// A function that gets an entry from the state.
(currentState = {}, [id]) => currentState[id],
// A function that sets an entry in the state.
(currentState = {}, [id], product) => ({
...currentState,
[id]: product
}),
// The loader function that fetches data. Should return a promise.
id => fetch(`/api/products/${id}`).then(response => response.json()),
// Optional: The expiration time for entries in milliseconds, beginning from each entry's load time, defaults to Infinity.
10 * 60 * 1000
);
createSingleEntryResource
Creates a resource that only has one entry. It conveniently supplies getter, setter, and predicate functions to createResource
under the hood, allowing you to simply supply a function that fetches your data. Once your data is fetched it will not be refetched.
const myResource = createSingleEntryResource(
// The loader function that fetches data. Should return a promise.
authToken => fetch(`/api/about_me?auth_token=${authToken}`),
// Optional: The expiration time for entries in milliseconds, beginning from each entry's load time, defaults to Infinity.
5000
);
createSingleEntryResource
example on CodeSandbox
createKeyedResource
Creates a resource that organizes its entries into an object literal. It takes a key setter function and a loader function that fetches your data. The key setter function derives the entry key from the same arguments that are supplied to the loader function. Once an entry is fetched it will not be refetched.
const myResource = createKeyedResource(
// A function that takes all of the arguments that are supplied to the loader, from resource.useEntry, and uses the returned value as the key
(authToken, userId) => userId,
// The loader function that fetches data. Should return a promise.
(authToken, userId) => fetch(`/api/users/${userId}?auth_token=${authToken}`),
// Optional: The expiration time for entries in milliseconds, beginning from each entry's load time, defaults to Infinity.
1 * 60 * 1000
);
createKeyedResource
example on CodeSandbox
persistResource
A higher order function that adds persistence to a specific resource.
const todosResource = persistResource(
// `getInitialState` - A function that returns a promise with the initial data. If the promise resolves, the data in the promise is used as the initial state of the resource. If the promise rejects, the load function of the resource will be called. This function is lazy and will only be called when an entry from the resource is requested.
() => {
const result = localStorage.getItem("todos");
return result ? Promise.resolve(JSON.parse(result)) : Promise.reject();
},
// `saveState` - A function that is called any time the resource state changes. It provides the new state, giving you the ability to persist it to something like `localStorage`.
state => {
localStorage.setItem("todos", JSON.stringify(state));
},
// `resource` - The resource to persist
createSingleEntryResource(() => fetch("/api/todos"))
);
RemoteResourceBoundary
Uses Suspense
under the hood to catch any thrown promises and render the fallback
while they are pending. Will also catch any errors that occur in the promise from a resource's loader and renderError
and call onLoadError
if provided.
const UserProfile = ({ userId }) => (
<RemoteResourceBoundary
/* A React node that will show while any thrown promises are pending. */
fallback={<p>Loading...</p>}
/* Optional: A callback that is invoked when any thrown promise rejects */
onLoadError={logError}
/* A render prop that receives the error and a function to clear the error, which allows the children to re-render and attempt loading again */
renderError={({ error, retry }) => (
<div>
<p>{error}</p>
<button onClick={retry}>Try again</button>
</div>
)}
>
<UserInfo userId={userId} />
<Tweets userId={userId} />
</RemoteResourceBoundary>
);
resource.useEntry
A React hook that takes a resource and an optional array of arguments and returns a tuple, very much like React's useState
. The second item in the tuple works like useState
in that it sets the in-memory state of the resource. Unlike useState
, however, the state is not local to the component. Any other components that are using the state of that same resource get updated immediately with the new state!
Under the hood react-remote-resource
implements a redux store. Every resource get its own state in the store.
import { createRemoteResouce, useAutoSave } from "react-remote-resource";
import { savePost, postsResource } from "../resources/posts";
const PostForm = ({ postId }) => {
const [post, setPost] = postsResource.useEntry(postId);
useAutoSave(post, savePost);
return (
<form
onSubmit={e => {
e.preventDefault();
savePost(post);
}}
>
<label>
Title
<input
type="text"
value={post.title}
onChange={({ target }) => {
setPost({ ...post, title: target.value });
}}
/>
</label>
<label>
Content
<textarea
value={post.title}
onChange={({ target }) => {
setPost({ ...post, content: target.value });
}}
/>
</label>
<button>Save</button>
</form>
);
};
This hook is very powerful. Let's walk through what happens when it is used:
createResource
) is used to get the entry out of the resource state.createResource
) is used to test whether or not the entry is valid. If not the loader (the fourth argument) will be invoked and the promise thrown.RemoteResourceBoundary
will handle the error. If the promise resolves, the setter function (the second argument give to createResource
) is used to set the resolved data in the resource state.useState
, will persist in memory. If a component unmounts and remounts the state will be the same as when you left it.
useAutoSave
A React hook that takes a value, a save function, and an optional delay in milliseconds (defaults to 1000). When the value changes the new value will be saved if the value remains the same for the delay time.
This is useful for optimistic UIs where the state of the resource is the source of truth and we are confident that the save will succeed.
import {
createRemoteResouce,
useEntry,
useAutoSave
} from "react-remote-resource";
import { savePost, usePost } from "../resources/posts";
const PostForm = ({ postId }) => {
const [post, setPost] = usePost(postId);
useAutoSave(post, savePost);
return (
<form
onSubmit={e => {
e.preventDefault();
savePost(post);
}}
>
<label>
Title
<input
type="text"
value={post.title}
onChange={({ target }) => {
setPost({ ...post, title: target.value });
}}
/>
</label>
<label>
Content
<textarea
value={post.title}
onChange={({ target }) => {
setPost({ ...post, content: target.value });
}}
/>
</label>
<button>Save</button>
</form>
);
};
useSuspense
A hook that takes a promise returning function. It will throw the returned promise as long as it is pending.
import { useSuspense } from "react-remote-resource";
import { saveUser } from "../resources/user";
const SaveButton = ({ onClick }) => (
<button onClick={useSuspense(onClick)}>Save</button>
);
const UserForm = () => (
<div>
...Your form fields
<Suspense fallback={<p>Saving...</p>}>
<SaveButton onClick={saveUser} />
</Suspense>
</div>
);
What are resource entries and how are they created?
An entry is data that is stored in the resource
when the promise of the load
finally resolves.
Each Resource Creator determines how the entries are organized and stored. As an example, createSingleEntryResource
creates a resource
that only has a single entry from an api.
const postsResource = createSingleEntryResource(() =>
fetch("/api/posts")
.then(normalizePosts)
.then(filterToCurrentUser)
.then(...)
);
The postsResource
above, will store the value as a single entry in the resource once all the .then
statements complete.
Does my data need to be in a specific shape to use
react-remote-resource
?
No, react-remote-resource
aims to organize your apis regardless of shape or type! Whatever resolves from the promise of the load
function will be stored as the entry.
When should I use
createKeyedResource
vscreateSingleEntryResource
?
The the main difference between createKeyedResource
and createSingleEntryResource
is how the created resource
stores its entries. Internally, every resource
organizes the data it receives from the completed load function into entries. A resource
's main concern is how the data in the entries should be available to your app, not necessarily how the external api is structured or called.
createKeyedResource
creates a resource with multiple entries that are organized by key. See the first parameter on how entries are keyed.
createSingleEntryResource
creates a resource with only a single entry. All data, regardless of the structure, will be stored in only one entry.
The following example scenerios show when createKeyedResource
and createSingleEntryResource
are best suited:
Assuming You have a users api that takes an id and returns user information:
/*
/api/users/:id
Example Response:
{
name: "name",
id: {userId},
...
}
*/
Scenerio 1:
The app only needs the current user's information. Since the app only needs one entry from the api (the current user's information and no others), createSingleEntryResource
would work well.
const load = id => fetch(`/api/users/${id}`);
const userResource = createSingleEntryResource(load);
const AboutMe = () => {
const [user] = userResource.useEntry(12345);
return ...;
};
Scenerio 2
The app needs to make individual requests to the same users API and store the results independently. createKeyedResource
would be the better choice.
const load = id => fetch(`/api/users/${id}`);
const usersResource = createKeyedResource(id => id, load);
const User = ({ id }) => {
const [user] = usersResource.useEntry(id);
return ...;
}
const UserList = ({ ids }) => {
return ids.map(id => <User key={id} id={id} />);
}
Assuming you have a clients API that takes an account_rep_id
and returns a list of clients for that specific account rep:
/*
/api/clients/:account_rep_id
[
{
id: 123454,
...
},
{
id: 508923,
...
},
{
id: 14,
...
},
{
id: 995,
...
}
]
*/
Scenerio 1:
The app only needs the current account rep's list: createSingleEntryResource
would work well.
const load = (id) => fetch(`/api/clients/${id}`);
const clientsResource = createSingleEntryResource(load);
const ClientList = () => {
const [clients] = clientsResource.useEntry(12345);
return ...;
};
Scenerio 2:
The app needs multiple account rep's client list: createKeyedResource
is better suited for this purpose.
const load = account_rep_id => fetch(`/api/clients/${account_rep_id}`);
const clientsResource = createKeyedResource(id => id, load);
const ClientList = ({ account_rep_id }) => {
const [clients] = clientsResource.useEntry(account_rep_id);
return ...;
}
You have a posts api that takes no parameters and returns a list of posts organized by post id:
/*
/api/posts
[
{
id: 882,
userId: 5
},
{
id: 622,
userId: 10
},
{
id: 622,
userId: 10
},
{
id: 1,
userId: 1
},
{
id: 102,
userId: 10
}
]
*/
The app itself needs the data to be organized by userId.
createSingleEntryResource
could be utilized and composed for this:
const postsResource = createSingleEntryResource(() =>
fetch("/api/posts").then(posts =>
posts.reduce(
(acc, post) => ({
...acc,
[post.userId]: acc[post.userId] ? [...acc[post.userId], post] : [post]
}),
{}
)
)
);
const useUsersPosts = id => {
const [allPosts] = postsResources.useEntry();
return allPosts[id] || [];
};
createSingleEntryResource
and createKeyedResource
are composed versions of createResource
. If they do not fit your use case, try using createResource
directly, as it may better suit your needs.
FAQs
Intuitive remote data management in React
The npm package react-remote-resource receives a total of 7 weekly downloads. As such, react-remote-resource popularity was classified as not popular.
We found that react-remote-resource demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Research
The Socket Research Team breaks down a malicious wrapper package that uses obfuscation to harvest credentials and exfiltrate sensitive data.
Research
Security News
Attackers used a malicious npm package typosquatting a popular ESLint plugin to steal sensitive data, execute commands, and exploit developer systems.
Security News
The Ultralytics' PyPI Package was compromised four times in one weekend through GitHub Actions cache poisoning and failure to rotate previously compromised API tokens.