-
Notifications
You must be signed in to change notification settings - Fork 560
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Context.write #89
Context.write #89
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,254 @@ | ||
- Start Date: 2018-11-19 | ||
- RFC PR: (leave this empty) | ||
- React Issue: (leave this empty) | ||
|
||
# Summary | ||
|
||
Proposes an extension to the context API for updating the default context value. This would allow for React-managed state that lives outside the UI tree and is shared across roots. | ||
|
||
# Basic example | ||
|
||
```js | ||
const Context = React.createContext(initialValue, contextDidUpdate); | ||
// Update the global context value across all roots. Any context consumer that | ||
// is not wrapped in a Provider will re-render with this value. | ||
Context.write(newValue); | ||
// Functional updates also supported for access to the previous value. | ||
Context.write(prevValue => newValue); | ||
|
||
function contextDidUpdate(newContext) { | ||
// Optional callback that fires whenever context changes | ||
} | ||
``` | ||
|
||
# Motivation | ||
|
||
## Caching external data | ||
|
||
Where should a React app store data fetched over IO? | ||
|
||
It may be tempting to cache it in component state. This is a common pattern for pre-Suspense React apps. But there are some drawbacks. For example, if a Comment component depends on data from the server, consider the implications of storing that data in the Comment component's local state. What if the same comment data is rendered by a different component on another part the page? I shouldn't have to fetch the same comment twice. What if I navigate away and my comment is unmounted, but then I later navigate back? I don't want to refetch the same comment again merely because the old instance was unmounted. The underlying principle here is that data fetched over IO typically does not semantically align with the lifetime of a component used to present that data. The data and its presentation are separate concepts. | ||
|
||
Still, although caching in component state isn't ideal, it's arguably "good enough" for many use cases today. | ||
|
||
Not so with Suspense. The Suspense model is that the first time React renders an IO-bound component, an exception is thrown when attempting to read the data. While React is waiting for the data to load, it continues rendering the rest of the tree, but it doesn't commit the result; the partially completed tree is _discarded_. Once the data has resolved, React attempts to render the entire tree over again from scratch. It may, as an optimiztion, reuse parts of the previous attempt. But it's not a guarantee. That means using local component state to cache data won't work, because that state will most likely not be persisted. The underlying priciple here is that you can't cache the intermediate results of a computation (server data) on the output of the computation itself (the React tree). | ||
|
||
One might argue our Suspense implementation is wrong and we should change it. However, the throw-it-out-and-try-again design is not primarily an implementation decision, but a modeling one. The main motivating use case is server rendering. If data were cached using the component tree, then the server would need to track the state of every component. By moving to an external cache, the React server renderer can remain stateless; all it needs to track is which parts of the tree have yet to complete. Caching on the component tree would also complicate client-side hydration. The server renderer would need to serialize the tree of component state and send it to the client. Not only is this complicated, it's bad practice. It's better to hydrate that data into in a normalized data cache, anyway, for the reasons described above. | ||
|
||
Ok, so storing in component state won't work with Suspense. Where then? React currently doesn't have a good answer to this question. | ||
|
||
### Concurrent invalidation | ||
|
||
Caching in a global singleton, or some other object that's not managed by React, will work up to a point. The hard part is once you need to update or invalidate the cache. | ||
|
||
Today, most frameworks that bind React to an external store (e.g. React Redux, Relay) are push-based. The store is mutated **first**, then a change event **pushes** the update to the affected React components. Because rendering is synchronous, the browser is blocked from painting until all the components have finished rendering. Subsequent updates and user inputs are also blocked. Because the whole transition is uninterruptible, the UI is never in an inconsistent state. Push-based architectures are common, but the main drawback is that they rely on sequential processing of updates. Once a state transition is in progress, it must finish completely, even if a higher priority event is received in the meantime. | ||
|
||
For component state, React solves this problem using **pull**-based state transitions. When React receives an update, it does not mutate the state of the component immediately, but instead adds the update to a priority queue. Later, React will **pull** those updates off the queue as it's rendering the next screen. The state is only mutated **after** the entire screen has finished rendering. Because nothing is mutated until the end, different priority levels can render concurrently. React can start rendering at a normal priority, pause somewhere in the middle, switch to a higher priority task, then resume the original task after. | ||
|
||
That's how React concurrently transitions **component** state, but we've already discussed why component state is not ideal for caching data. | ||
|
||
What we need is a way to 1) store data outside of the component tree, 2) without creating inconsistencies in the UI (tearing), and 3) in a way that takes full advantage of React's concurrent rendering mode. | ||
|
||
The most crucial detail for acheiving 2) is **where the mutation occurs**. To avoid tearing, the store cannot be mutated until React has finished rendering. | ||
|
||
To achieve 3), pending updates must be stored in a priority queue, and React components must be able to read from that queue at different priority levels. | ||
|
||
## Automatic dependency injection | ||
|
||
One of the most common uses of context in React is for dependency injection. Instead of relying on global mutable state, a framework may broadcast a value to a subtree using the context API. This preserves the option to wrap a tree of components in a nested context provider without changing the consumers. In practice, this means many React apps have a section near the root that contains all the provider components needed to render the app. Not only is this tedious, but it also means React has to load all the code for those libraries up front, even if the consumers haven't mounted yet. | ||
|
||
The new context API partially addresses this problem by allowing for a default context value. Any consumer that reads from context but is not wrapped in a provider receives the default value. But there's no mechanism to update that value, so it's insufficient for any type of context that is stateful. The only way to do this today is to wrap the app in a stateful component and pass state to a context provider. | ||
|
||
## Sharing state across roots | ||
|
||
Wrapping a tree with a stateful context provider component doesn't address apps that are comprised of multiple roots. Each root needs its own provider. If the data is stored locally in the provider component, then each root will have a separate cache, leading to duplicate requests. If you move the state outside of the providers to an external store, then the providers will need to subscribe to the store's updates. | ||
|
||
An external store's state is also not managed by React, so it can't be fully compatible with concurrent rendering. Short of full compatibility, even limited compatibility without tearing is difficult to implement correctly without deep knowledge of React's rendering model. An idiomatic API designed for this use case should at the very least make it possible to lift state out of React without causing tearing and without always falling back to synchronous mode. | ||
|
||
# Detailed design | ||
|
||
## Motivating examples | ||
|
||
### Immutable store (like React Redux) | ||
|
||
Implementing immutable stores is straightfoward. Actions are applied in the render phase using the functional form of `Context.write`. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There's a point I'm misunderstanding. What does it mean concretely that Actions are applied in the render phase using the functional form of There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Explanation of render phase versus commit phase: https://twitter.com/acdlite/status/977291318324948992 |
||
|
||
```js | ||
const Store = React.createContext(initialState); | ||
|
||
export function dispatch(action) { | ||
Store.write(state => reducer(state, action)); | ||
} | ||
|
||
export function useStore() { | ||
return useContext(Store); | ||
} | ||
``` | ||
|
||
### Mutable store (like React Cache) | ||
|
||
Mutable stores are a bit trickier but can still work. The trick is to avoid mutating the store directly. Instead, add pending mutations to a queue, and only flush the queue once you reach the commit phase (using the `contextDidUpdate` callback). | ||
|
||
In this example, the context value for the cache is comprised of two maps: a cached map, and pending map. Components read from the pending map first, before falling back to the cached map. The pending map is immutable: it's populated by `Context.write`. The cached map is mutable: it's mutated in the commit phase. | ||
|
||
The advantage of this approach is that only the pending map needs to be immutable (copy-on-write). Because the pending map is a small subset of the entire cache, this minimizes the amount of copying needed to support concurrent access. | ||
|
||
```js | ||
const Cache = React.createContext( | ||
{cachedRecords: null, pendingRecords: null}, | ||
cacheDidCommit, | ||
); | ||
|
||
export function read(key) { | ||
const cache = Cache.read(); | ||
const {cachedRecords, pendingRecords} = cache; | ||
|
||
let record; | ||
if (pendingRecords !== null && pendingRecords.has(key)) { | ||
// Always read from pending map first | ||
record = pendingRecords.get(key); | ||
} else if (cachedRecords !== null && cachedRecords.has(key)) { | ||
// If there's no pending update, read from cached map. | ||
record = cachedRecords.get(key); | ||
} else { | ||
// If there's no match, create a new, empty record. | ||
record = {tag: 'pending', value: null}; | ||
} | ||
|
||
switch (record.tag) { | ||
case 'pending': | ||
// This initiates a request and throws a promise to suspend the render. | ||
suspendOnPendingRecord(cache, record); | ||
case 'resolved': | ||
// Return the cached value. | ||
return record.value; | ||
case 'rejected': | ||
// Throw an error. | ||
throw record.value; | ||
} | ||
} | ||
|
||
export function invalidateByKey(key) { | ||
// Create an empty record. | ||
const newRecord = {tag: 'pending', value: null}; | ||
|
||
// Schedule an update to overwrite the cached record with the new one. | ||
Cache.write(cache => { | ||
const {cachedRecords, pendingRecords} = cache; | ||
if (cachedRecords === null || !cachedRecords.has(key)) { | ||
// If there's not already a cached value, then there's nothing to | ||
// invalidate. Reuse the existing cache. | ||
return cache; | ||
} | ||
// Add to the pending records map. This needs to be an immutable operation: | ||
// we copy the previous map before setting. The cache itself isn't updated | ||
// until the commit phase. | ||
pendingRecords = new Map(pendingRecords); | ||
pendingRecords.set(key, newRecord); | ||
return {cachedRecords, pendingRecords}; | ||
}); | ||
} | ||
|
||
export function invalidateAll() { | ||
// Clear the entire cache. | ||
Cache.write({cachedRecords: null, pendingRecords: null}); | ||
} | ||
|
||
function cacheDidCommit(committedCache) { | ||
// Now that we've reached the commit phase, it's safe to mutate. Collapse the | ||
// two maps into one by overwriting the cached map with the values from the | ||
// pending map. | ||
const pendingRecords = committedCache.pendingRecords; | ||
if (pendingRecords !== null) { | ||
const cachedRecords = committedCache.cachedRecords; | ||
if (cachedRecords === null) { | ||
cache.cachedRecords = pendingRecords; | ||
} else { | ||
pendingRecords.forEach((record, key) => { | ||
cachedRecords.set(key, record); | ||
}); | ||
} | ||
// The pending records have been persisted, so we no longer need them. | ||
cache.pendingRecords = null; | ||
} | ||
} | ||
``` | ||
|
||
# Drawbacks | ||
|
||
## How to handle multiple roots | ||
|
||
The trickiest question is how to deal with multiple roots. React does not guarantee consistency across roots; each root has its own commit phase, and suspending inside one root has no effect on the others. This isn't observable in synchronous mode, since React will block the main thread (including paint) until every root's commit phase has finished. In concurrent mode, however, React may yield in between each commit. With Suspense, the time between each commit phase could vary by many seconds. | ||
|
||
This model has several consequences. For each context, React would have to maintain a separate version per root, as well as a separate queue of updates. The `contextDidUpdate` callback complicates this further. If each root updates separately, it only makes sense to fire the callback once every root has committed, which necessitates reference counting or some similar tracking mechanism. | ||
|
||
An alternative model is to treat all the roots as siblings and commit them at the same time. This could be viable for single page React apps (where there aren't that many roots, anyway). It doesn't work so well in cases where React is embedded inside another framework (e.g. progressive enhancement of a server rendered app), where temporary inconsistencies may be desirable. For example, the opt-in API for concurrent mode relies on roots committing separately so that you can upgrade some roots to concurrent mode without upgrading the entire app. In an app with mixed synchronous and concurrent roots, a unified commit would mean that every call to `Context.write` has to be synchronous, which probably makes this option a non-starter. | ||
|
||
In either of these models, React would need to track a global list of all roots in order to schedule updates on them. This isn't something we do currently, and it means roots would no longer be automatically garbage collected; discarded roots would need to be explicitly unmounted to remove them from the global list. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If by "discarded roots" you mean those which are just removed from DOM without calling There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How would that work? AFAIK MutationObserver only works for immediate children. It won't tell you if a tree is disconnected by an ancestor node. |
||
|
||
What's clear from exploring this issue and others is that there are significant implementation costs to supporting multiple roots. In the future, we may move to a portal-first API that enforces a single root by default, and extract support for multiple roots to a separate package. | ||
|
||
### Discordance with Hook API | ||
|
||
`Context.write` has a similar API to the `useState` Hook, and `contextDidUpdate` is similar to `useEffect`. Perhaps we could consider an alternate proposal that uses these Hooks directly. See to the "Alternatives" section for an example. | ||
|
||
# Alternatives | ||
|
||
## Do nothing and leave the problem to user space | ||
|
||
We've come this far without the need for an API like this. But the main reason we need to address this now is because of Suspense and concurrent rendering. Given how important this use case is for Suspense, doing nothing seems unlikely. | ||
|
||
## Mount a "shell" component and cache using local state | ||
|
||
Another way to move this to user space would be to have a shell component at the root of the app that initially renders with no children. On mount, it schedules a re-render to mount the children. Because the shell is already mounted, the children can cache values in the shell's local state. | ||
|
||
This doesn't address the multiple root problem, nor does it work with React's server renderer, which does not have updates. | ||
|
||
## Use Hooks | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. +1 for more symmetry between component state and context. Ideally, lifting component state & logic up to into a shared context wouldn't require rewriting the whole thing. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Agreed. I've been doing stuff on this direction on https://github.com/diegohaz/constate Ideally, we should be writing local state until we really need to lift it up. Then, refactoring it into global/shared/contextual state should be something really simple. |
||
|
||
Leverage Hooks for updates and effects, instead of adding new APIs that do a subset of the same thing. Here's an example that logs whenever the theme changes: | ||
|
||
```js | ||
// This needs to be a new method because using `createContext` would be a | ||
// breaking change. But the opaque return type is the same. | ||
const Theme = React.createGlobalState(ref => { | ||
const [theme, setTheme] = useState('light'); | ||
useEffect( | ||
() => { | ||
console.log('Theme changed: ' + theme); | ||
}, | ||
[theme], | ||
); | ||
|
||
useImperativeMethods(ref, () => ({setTheme})); | ||
|
||
return theme; | ||
}); | ||
|
||
export function useTheme() { | ||
return useContext(Theme); | ||
} | ||
|
||
export function toggleTheme() { | ||
Theme.current.setTheme(theme => { | ||
return theme === 'light' ? 'dark' : 'light'; | ||
}); | ||
} | ||
``` | ||
|
||
A more realistic example is a router that sets up a global `'popstate'` listener. | ||
|
||
Leveraging Hooks would have several advantages: | ||
|
||
- Smaller API surface area. (Excepting the need for a separate `createContext` and `createGlobalState` APIs, though we could unify them in a major release.) | ||
- Allows you to move or reuse code between component state and global state with minimal changes. | ||
|
||
(I thought about moving this to a separate proposal, but it doesn't seem sufficiently different from `Context.write` to merit its own document. The bulk of the proposal is the same.) | ||
|
||
# Adoption strategy | ||
|
||
TODO | ||
|
||
# How we teach this | ||
|
||
TODO |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
"The underlying principle"