-
-
Notifications
You must be signed in to change notification settings - Fork 3.4k
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
React-Redux Roadmap: v6, Context, Subscriptions, and Hooks #1177
Comments
I'm not really sure the It would allow react-redux to move the bailout to the I still have some ideas for things to try without breaking API, though. |
I'm open to experimenting with as many variations on implementation approaches as we can come up with :) All the more reason to have additional tests that can quantify and express the additional constraints we need to meet, beyond what we have in the repo now. |
Thank you!! |
FWIW I don’t see anything preventing us from adding provider support to shallow renderer. If that’s the concern about testing. |
@Jessidhia The batchedUpdates thing is more about fixing the “zombie child”. It’s the idiomatic React solution for always traversing from the top. It can also improve perf outside event handlers but avoiding the bad ordering is the main motivation. |
Please don't forget to support immutable data structure libraries like immutable-js. They have a special equality check like is() and we need a way to define a custom comparison function for |
This is a fantastic writeup, sincerely thank you so much for your hard work @markerikson! 🙌
I do agree with this. That said, this will technically break some implementations of dynamically injected reducers in v6 since they currently have to rely on those private APIs, so I was glad to see you explicitly mention that issue. 😀 I don't think any of the "big ones" are using I really agree with a test-based approach here, if I can find the time I'll try to contribute some tests focusing on SSR. |
Just wanted to say I'm excited about this roadmap and feel it's the correct direction forward until both context performance for large stores with frequent updates and bailout of useContext is added/supported. I'd love to see a compilation of all the hooks approaches the community has developed thus far and review in a centralized place! Awesome work @markerikson , the community greatly appreciated your work! |
@vincentjames501 : yup, agreed. A couple months ago, @adamkleingit put together this list of existing Redux hooks libs: https://twitter.com/adamklein500/status/1072457324932067329 Collating that list of existing Redux hooks libs is absolutely something that someone else besides me can do, and would really help (like, a Google Docs spreadsheet or something like that). Links, summaries, API comparisons, etc. Others not in that list as I run across them: |
It's really interesting to me that you mention using setState batching as a potential solution because that's the way I originally solved the zombie/tearing issue for react-redux in the #99 PR. I guess things can come around sometimes :) That PR didn't actually use the In the @gaearon's "React as a UI Runtime" article he explains how and why React uses batching for event handlers https://overreacted.io/react-as-a-ui-runtime/#batching This really resonates with me because the reason for it seems to be basically same as why Redux would need it. And using it the issue is pretty easy to solve. Inspired by this I created my own redux-hooks implementation that does not suffer from the zombie issue: https://github.com/epeli/redux-hooks It works by using static context which contains the store reference and an array reference. When the Here's a simple todoapp using this implementation Codesandbox: https://codesandbox.io/s/github/epeli/typescript-redux-todoapp/tree/hooks I really cannot comment about the performance of this since I just put it together but I don't think there's any reason why it couldn't be good. This implementation now just does the basic performance optimization by bailing out if mapState does not produce new value (shallow equal check). I'd be super interested hearing any feedback on this. Even just for the internal hooks usage. This is the first thing I've ever written with hooks. EDIT: Oh, and this not a hooks specific way to implement this. The same method would work for the regular HOC |
@markerikson ❤️ To weigh in slightly: for Here was my initial performance investigations of the new context api for propigating updates: facebook/react#13739 |
Hi everyone! I just migrated one of my libs (which has a pretty extensive test coverage) to use
I hope these help a bit. |
I've done some early experimenting the last couple evenings with trying to rework the guts of https://github.com/reduxjs/react-redux/tree/connect-subscription-experiments A few notes:
FWIW, this is just me having hacked on stuff for a couple evenings - I'm just throwing out what I've tried and thought of so far. |
Great work, I will try to dive deeper into it when I can, but I immediately had one question. Since Haven’t checked your code or any details so might well be missing something which is why I’m asking. Might try to play with this tomorrow. :) |
No, the v6 implementation specifically supports passing a custom context instance as a prop directly to a connected component, regardless of whether or not you also passed a custom context instance as an option when calling react-redux/src/components/connectAdvanced.js Lines 245 to 256 in ab77450
This was primarily intended to act as an alternative for the removal of the " |
Thanks for explaining, I had somehow missed that part. Certainly makes it trickier. :/ |
Using context from props does indeed seem like it necessarily adds another wrapper if not using hooks. I guess one approach to handle it with backwards compat might be to check if context-prop exists, not set up subscription in that case and render extra wrapper only then (+handle prop-change case..). Seems tricky and possibly verbose though. Good performance in default case or custom context via |
Could explain a little or point to a test of this "children first" scenario?
I'd like to take a look at it. |
Back in December, we released React-Redux version 6.0. This release featured a major rearchitecture of the implementations for
<Provider>
andconnect
, centered around our use of React's context API.We put the pre-release builds through extensive tests, including our existing unit test suite and multiple performance benchmark scenarios. We also requested early adopter feedback on beta versions. Based on all that information, we determined that v6 was ready for public release, and published it.
I think most library maintainers would agree that the best way to guarantee getting bug reports is to publish a new major version, because A) that means the entire community is about to start using the new code, and B) it's impossible for a maintainer to anticipate every way that the library is being used. That's definitely been true with React-Redux version 6.
Version 6 does work as intended, and many of our users have successfully upgraded from v5 to v6, or started new projects with v6. However, we've seen a number of concerns and challenges that have come up since v6 was released. These relate to both the current behavior of v6 and its API, as well as future potential changes such as the ability to ship official React hooks that use Redux.
In this issue, I want to lay out a description of the changes we made in v6 and why we made them, the challenges we are currently facing, the constraints that potential solutions need to conform to, and some possible courses of action for resolving those challenges.
TL;DR:
createContext
useRedux()
hooks that rely on contextconnect
may require a major version bump.Implementation Changes in Version 6
In my blog post The History and Implementation of React-Redux, I described the technical changes we made from v5 to v6, and why we made them. Summarizing those changes:
store
, and use that instead of the instance from contextcreateContext
API<Provider>
subscribes to the store - the components just read the store state from contextstore
as a prop is meaningless and was removedWe made these changes for several reasons:
Challenges with v6
Performance
During the development of v6, we put together a performance benchmarks suite that we could use to compare the behavior of different builds of React-Redux. These benchmarks are artificial stress tests that don't necessarily match real-world app setups, but they at least provide some consistent objective numbers that can be used to compare the overall behavior of builds to keep us from accidentally shipping a major performance regression.
Our comparisons showed that v6 was generally slower than v5 by different amounts in different scenarios, but we concluded that real-world apps would probably not experience any meaningful differences. That seems to have been true for most users, but we have had several reports of performance decreases in some apps.
Fundamentally, this is due to how v6 relies on context for propagating state updates. In v5, each component could run its own
mapState
function, check the results, and only callthis.setState()
if it knew it needed to re-render. That meant React only got involved if a re-render was truly necessary. In v6, every Redux state update immediately causes asetState()
in<Provider>
at the root of the tree, and React always has to walk through the component tree to find any connected components that may be interested in the new state. This means that v6 ultimately results in more work being done for each Redux store update. For usage scenarios with frequent Redux store updates, this could result in potential slowdowns.store
as PropWe removed the ability to pass a prop named
store
to connected components specifically because they no longer subscribe to stores directly. This feature had two primary use cases:<Provider>
Our general guidance for the first use case was to always wrap components in
<Provider>
in tests. We tried to provide a solution for the second use case by allowing users to pass custom context instances to both<Provider>
and connected components.However, since the release of v6, we've had several users express concerns that the removal of
store
as a prop breaks their tests, and that there are specific problems with trying to use the combination of Enzyme'sshallow()
function with a<Provider>
(and React's new context API in general).Context API Limitations
At first glance, React's new
createContext
API appears to be perfectly suited for the needs of a library like React-Redux. It's built into React, it was described as "production-ready" when it was released, it's designed for making values available to deeply-nested components in the tree, and React handles ordering the updates in a top-down sequence. The<Context.Provider>
usage even looks very similar to React-Redux's<Provider>
.Unfortunately, further usage has shown that context is not as well suited for our use case as we first thought. To specifically quote Sebastian Markbage:
React-Redux and Hooks
In addition to concerns about performance and state update behaviors, the initial release of the React Hooks API will not include a way to bail out of updates triggered by a context value update. This effectively blocks React-Redux from being able to ship some form of a
useRedux()
hook based on our current v6 implementation.To again quote Sebastian:
I'll pull out one specific sentence there for emphasis:
Constraints
Whatever solutions we come up with for these challenges need to fit within a variety of overlapping constraints.
Performance Should Match or Improve vs v5
Ideally, v6 should be at least as fast as v5, if not faster. "Faster", of course, is entirely dependent on what metrics we're measuring, and how we're measuring them.
Handle Use Cases for
store
as PropWe need to support the use cases that were previously handled by passing a store directly as a prop to connected components. As part of that, we should ensure that we have tests that cover these usages.
Future React Compatibility
We have some idea what the potential concerns are around React's future Concurrent Mode and Suspense capabilities, but it would help to have some concrete examples that we can use to ensure we're either not breaking application behavior, or at least can help us quantify what the potential breakages are.
Quoting Dan Abramov:
At a minimum, we should ensure that React-Redux does not cause any warnings when used inside a
<StrictMode>
component. That includes use of semi-deprecated lifecycle methods likecomponentWillReceiveProps
(which was used in v5).Don't Re-Introduce "Zombie Child" Problems
Up through v4, we had reports of a bug that could happen when children subscribed before parents. At a technical level, the actual issue was:
ownProps
with new state inmapStateToProps()
mapStateToProps()
for a component that will be unmounted later in the overall render cycle, combined with a failure to handle cases where the values needed from the store might not existAs an example, this could happen if:
mapState
then ran before the parent had a chance to re-render without that childmapState
tried to read nested state without safely checking to see if that data existed firstv5 specifically introduced an internal
Subscription
class that caused connected components to update in a tiered approach, so that parents always updated before children. We removed that code in v6, because context updates top-down already, so we didn't need to do it ourselves.Whatever solutions we come up with should avoid re-introducing this issue.
Allow Shipping Redux Hooks
The React community as a whole is eagerly awaiting the final public release of React Hooks, and the React-Redux community is no exception. Redux users have already created a multitude of unofficial "Redux hooks" implementations, and have expressed a great deal of interest in an official set of hooks-based APIs as part of React-Redux.
We absolutely want to ship our own official Redux hooks as soon as possible. Whatever changes we decide on need to make that feasible.
Potentially Use Hooks in
connect
When hooks were announced, I immediately prototyped a proof of concept that reimplemented
connect
using hooks internally. That simplified theconnect
implementation dramatically.I'd love to use hooks inside React-Redux itself, but that would require bumping our peer dependency on React from the current value of 16.4 in v6, to a minimum of 16.8. That would require a corresponding major version bump of React-Redux to v7.
That's a potential option, but I'd prefer not to bump our own major version if we can avoid it. It should be possible to ship a
useRedux()
as part of our public API as a minor 6.x version, and leave it up to the user to make sure they've got a hooks-capable version of React if they want to import that hook. Then again, it's also possible that a hooks-based version ofconnect
would be necessary to solve the other constraints.Continue to Work with Other Use Cases
The broader React-Redux ecosystem has updated itself to work with v6. Some libraries have had to change from using the
withRef
option toforwardRef
. Other libraries that were accessing the store out of the (undocumented private) legacy context have switched to accessing the store out of our (still private)ReactReduxContext
instance.This has also brought up other semi-niche use cases that we want to support, including having
connect
work with React-Hot-Loader, and support for dynamically updating reducers and store state in SSR and code-splitting scenarios.The React-Redux community has built some great things on top of our baseline capabilities, and we want to allow people to continue to do so even if we don't explicitly support everything ourselves.
Courses of Action
So here's where the rubber meets the road.
At the moment, we don't have specific implementations and solutions for all those constraints. We do have some general outlines for some tasks and courses of action that will hopefully lead us towards some working solutions.
Switch
connect
Back to Direct SubscriptionsBased on guidance from the React team, the primary thing we should do at this point is switch
connect
back to using direct subscriptions.Unfortunately, we can't just copy the v5 implementation directly into the v6 codebase and go with it as-is. Even ignoring the switch from legacy context to new context, v5 relied on running memoized selectors in
componentWillReceiveProps
to handle changes to incoming props, and then returningfalse
inshouldComponentUpdate
if necessary. That caused warnings in<StrictMode>
, which we want to avoid.We need to design a new internal implementation for store subscription handling that satisfies the listed constraints. We've done some early experiments, but don't have any specific approaches yet that we can say are the "right" way to do it.
We do actually already put the store instance into
createContext
, so nothing needs to change there. The specific values we put into context are not considered part of our public API, so we can safely remove thestoreState
field from context.Bringing back direct subscriptions does mean that we can probably bring back the ability to pass
store
as a prop directly to connected components. That should hopefully resolve the concerns about testing and isolated alternate-store usage, because it's the same API that solved those use cases previously.Expand Test Suite for More Use Cases
We currently have a fairly extensive unit test suite for
connect
and<Provider>
. Given the discussions and issues we're facing, I think we need to expand that suite to make sure we're better covering the variety of use cases the community has. For example, I'd like to see some tests that do Enzyme shallow rendering of connected components, mock (or actual) SSR and dynamic loading of slice reducers, and hopefully tests or apps that show the actual problems we might face in a Concurrent Mode or Suspense environment.Consider Marking Current Implementation as Experimental
The v6 implementation does work, and there may be people who prefer to use it. Rather than just throw it away, we could potentially keep it around as a separate entry point, like
import {connect, Provider} from "react-redux/experimental"
.Improve Benchmarks Suite
Our current benchmarks are somewhat rudimentary:
I would really like our benchmarks to be improved in several ways:
mapState
functions, queuing updates, wrappers re-rendering, etc).unstable_batchedUpdates
API, to see how much of a difference that makes in overall performanceIn general, we need to better capture real-world behavior.
Officially Support Batched React Updates
ReactDOM has long included an
unstable_batchedUpdates
API. Internally, it uses this to wrap all event handlers, which is why multiplesetState()
calls in a single event handler get batched into a single update.Although this API is still labeled as "unstable", the React team has encouraged us to ship an abstraction over this function officially as part of React-Redux. We would likely do this in two ways:
batch
that can be used directly by end users themselves, such as wrapping multiple dispatches in a thunkReactReduxEnhancer
that would wrap dispatches inunstable_batchedUpdates
, similar to howtappleby/redux-batched-subscribe
works.It's not yet clear how much this would improve overall performance. However, this may hopefully act as a solution to the "zombie child component" problem. We need to investigate this further, but the React team has suggested that this would be a potential solution.
Currently, ReactDOM and React Native apparently both separately export their own versions of
unstable_batchedUpdates
, because this is a reconciler-level API. Since React-Redux can be used in either environment, we need to provide some platform abstraction that can determine which environment is being used, and import the method appropriately before re-exporting it. This may be doable with some kind of.native.js
file that is picked up by the RN build system. We might also need a fallback in case React-Redux is being used with some other environment.Hooks
We can't create
useRedux()
hooks for React-Redux that work correctly with v6's context-based state propagation, because we can't bail out of updates if the store state changed but themapState
results were the same. However, the community has already created numerous third-party hooks that rely on direct store subscriptions, so we know that works in general. So, our ability to ship official hooks is dependent on us first switching back to direct store subscriptions inconnect
, because bothconnect
and any official hooks implementation need to share the same state propagation approach.There's a lot of bikeshedding that can be done about the exact form and behavior of the hooks we should ship. The obvious form would be to have a direct equivalent of
connect
, likeuseRedux(mapState, mapDispatch)
. It would also be reasonable to have separate hooks likeuseMapState()
anduseMapDispatch()
. Given the plethora of existing third-party hooks libs, we can survey those for API and implementation ideas to help determine the exact final APIs we want to ship.In theory, we ought to be able to ship these hooks as part of a 6.x minor release, without requiring that you have a minimum hooks-capable version of React. That way, users who are still on React <= 16.7 could conceivably use React-Redux 6.x, and it will work fine as long as they don't try to actually use the hooks we export.
Long-term, I'd probably like to rework
connect
to be implemented using hooks internally, but that does require a minimum peer dependency of React 16.8. That would be a breaking change and require a major version bump for React-Redux. I'd like to avoid that on general principle. But, if it turns out that a hooks-basedconnect
implementation is actually the only real way to satisfy the other constraints, that may turn out to be necessary.Requests for Community Help
There's a lot of stuff in that list. We need YOUR help to make sure React-Redux works well for everyone!
Here's how you can help:
connect
, including experimenting with implementations yourself, and giving us feedback on any test releases we publish.We can start with some initial discussion in this issue, but I'll probably try to open up some specific issues for these different aspects in the near future to divide up discussion appropriately.
Final Thoughts
There's a lot of great things ahead for React. I want to make sure that React and Redux continue to be a great choice for building applications together, and that Redux users can take advantage of whatever capabilities React offers if at all possible.
The React-Redux community is large, growing, smart, and motivated. I'm looking forward to seeing how we can solve these challenges, together!
The text was updated successfully, but these errors were encountered: