fix(deps): update dependency @reduxjs/toolkit to v2 - autoclosed
This MR contains the following updates:
Package | Type | Update | Change |
---|---|---|---|
@reduxjs/toolkit (source) | dependencies | major | ^1.9.5 -> ^2.0.0 |
Release Notes
reduxjs/redux-toolkit
v2.2.5
This bugfix release fixes an issue in the recent createEntityAdapter
sorting perf improvements that could (in specific cases) cause Immer to throw an error when trying to read a plain JS value instead of a proxy-wrapped value.
What's Changed
- Fix missed spot where use of
current
may fail if the value is not a draft by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/4412
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.2.4...v2.2.5
v2.2.4
This bugfix release improves sorting performance in createEntityAdapter
, shrinks the code size in matcher utilities, fixes assorted issues with query hooks, and makes several TS tweaks.
Changelog
Entity Adapter Sorting Performance
Users reported in #4252 that the sorting performance of createEntityAdapter
seemed abnormally bad - the provided comparison functions were being called far more times than expected.
Upon investigation, we had a couple of problems. We were always starting from an array that was in insertion order, not the existing sorted order, and that would always require significant effort to re-sort even if there weren't any actual changes to the sorted results. Also, the sorting checks required frequent access to Immer's Proxy-wrapped values, even in cases where all we needed was the plain state values for comparison purposes.
We've reworked the internal sorting logic to always start from the existing sorted array, do reads against a plain value to avoid the Proxy getter overhead where possible, and optimized inserts into existing sorted arrays. This should significantly speed up sorted entity adapter behavior.
Matcher Code Size Optimization
We've reworked the internals of the thunk-related matchers to deduplicate some of the logic, shaving a few bytes off the final bundle size.
RTK Query Hook Updates
defaultSerializeQueryArgs
can now handle BigInt
values safely.
The isLoading
flag logic was improved to handle errors when a query hook tries to subscribe.
TS Updates
create.asyncThunk
's types were improved to avoid cases where it might infer any
.
We've made several internal types changes to work correctly with React 19's upcoming types.
The retryCondition
method now receives unknown
as an argument, instead of always assuming the user is using fetchBaseQuery
.
Other Changes
The Reselect dep has been bumped to 5.1.0 to match the expected internal usage of createSelector.withTypes()
.
What's Changed
- Update remaining
Context
references to match the new nullableContext
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4336 - types: export
GetThunkAPI
type by @shrouxm in https://github.com/reduxjs/redux-toolkit/pull/4289 - Inline the deprecated
React.ReactChild
type by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4382 - Replace usage of deprecated
JSX
global namespace withReact.JSX
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4381 - Fix
useRef
usages to be called with an explicit argument ofundefined
. by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4380 - Bump
reselect
dependency to 5.1.0 to resolve #4200 by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4400 - Improve sorted entity adapter sorting performance by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/4361
- Prevent inference of any in create.asyncThunk by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4389
- try out simplifying matcher code by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4386
- Handle bigint in useQuerySubscription by @rkofman in https://github.com/reduxjs/redux-toolkit/pull/4315
- fix: ensure hook subscription failures do not reset isLoading state by @smacpherson64 in https://github.com/reduxjs/redux-toolkit/pull/4364
- Improve usability of the
initiate()
action for mutations. by @jared-ca in https://github.com/reduxjs/redux-toolkit/pull/4337 - Make retryCondition receive unknown error, as the user may not be using fetchBaseQuery by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3957
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.2.3...v2.2.4
v2.2.3
This bugfix release fixes the types for functions that accept a React Context instance to match the changes in React Redux v9.
What's Changed
- Update React Redux dependency to v9, and update docs to use
.withTypes
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4308
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.2.2...v2.2.3
v2.2.2
This patch release fixes an incorrect build setting for the legacy-esm
artifacts, and fixes an issue with RTKQ query hooks didn't always remove the cache entries if arguments were changed rapidly.
Changes
legacy-esm
Artifact Transpilation
The legacy-esm
build artifacts are intended for use by Webpack 4. Those were supposed to be transpiled to target "es2017"
, but were in fact still set to target "esnext"
- an oversight during the 2.0 development cycle. This release fixes that setting, so those artifacts are now correctly transpiled.
Other Fixes
RTKQ query hooks now handle additional actions around argument changes that should result in cache entries being removed.
Additionally, 2.2.1 contained a fix to an incorrectly named type: TypedUseMutationTrigger
is now TypedMutationTrigger
.
What's Changed
- rename TypedUseMutationTrigger to TypedMutationTrigger, and add deprecated alias by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4204
- Fixed memory leak in rapid hook arg changing by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4268
- Fix incorrect legacy-esm target by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/4284
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.2.0...v2.2.2
v2.2.1
v2.2.0
This minor release:
- Adds a second parameter to
entityAdapter.getInitialState(additionalProps, entities)
to allow prefilling state- Equivalent to
entityAdapter.setAll(entityAdapter.getInitialState(additionalProps), entities)
- First parameter can be
undefined
if no additional properties are desired
- Equivalent to
- Allows initialising
combineSlices
with no static reducers- Previously
const combinedReducer = combineSlices().withLazyLoadedSlices<LazyLoadedSlices>()
would have thrown an error - Now returns a "no-op" reducer that just returns an empty object until first reducer injected
- Previously
- Allows a new
'throw'
value foroverrideExisting
ininjectEndpoints
, which throws an error if a definition is injected with a name which is already used - Exports more type helpers for RTKQ hook and trigger types
- Exports types related to overriding result types in
enhanceEndpoints
- Fixes state inference for injected slices when undeclared (i.e. not in
LazyLoadedSlices
) - Adds a
action.meta.arg.isPrefetch
value to query thunk actions when prefetched
What's Changed
- Revamp type tests setup by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4095
- Bump Prettier and Prettier related packages by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4126
- Fix codemods to work with TypeScript 4.7+ by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4081
- Export types related to overriding the result types by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4134
- Migrate type tests to Vitest by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4127
- Fix RetryOptions type test by flipping order by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4136
- Format all files by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4135
- [Docs/Website] skipPollingIfUnfocused added to polling overview and query options by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4131
- [Docs] Changed create.asyncThunk setup admonition to caution by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4150
- Temporarily revert node-fetch bump until ESM issues are sorted by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4151
- Contributing Markdown Refresh by @itz-Me-Pj in https://github.com/reduxjs/redux-toolkit/pull/4139
- investigate re-adding size limit action by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4083
- Add size limit imports for more commonly used RTK exports by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4155
- pure some things by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4157
- cut back on suffixes tested for size by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4160
- Incorrect builder method referenced by @kantbtrue in https://github.com/reduxjs/redux-toolkit/pull/4161
- Rename
cli.js
tocli.mjs
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4169 - Migrate Codegen OpenAPI's unit tests to Vitest by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4137
- Fix wrong state for injected slices when not declared via
withLazyLoadedSlices
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4172 - [Docs, createEntityAdapter API]: add missing setOne and setMany signature by @kyselberg in https://github.com/reduxjs/redux-toolkit/pull/4173
- Corrected the "Observing cache behaviour" example point no.4 by @721-atikshaikh in https://github.com/reduxjs/redux-toolkit/pull/4174
- feat: add isPrefetch property in query action by @juliengbt in https://github.com/reduxjs/redux-toolkit/pull/4177
- allow initialising combined slice reducer with no static slices by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4184
- Create more Typed wrappers for RTKQ hook types by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4147
- proposal fix for axios base query types by @smff in https://github.com/reduxjs/redux-toolkit/pull/4186
- Provide 'throw' option for
overrideExisting
by @ffluk3 in https://github.com/reduxjs/redux-toolkit/pull/4189 - Use vite-tsconfig-paths to make path aliasing easier by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4175
- Remove trailing commas in
tsconfig.json
files of all CodesandBox examples by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4190 - [Docs] Added withTypes documentation in createDraftSafeSelector by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4143
- Add second parameter to getInitialState to prefill entities by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4183
New Contributors
- @itz-Me-Pj made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4139
- @kantbtrue made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4161
- @kyselberg made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4173
- @721-atikshaikh made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4174
- @smff made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4186
- @ffluk3 made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4189
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.1.0...v2.2.0
v2.1.0
This minor release:
- adds withTypes methods to
listenerMiddleware
andcreateDraftSafeSelector
- adds a
skipPollingIfUnfocused
option to RTK Query - adds the ability to customise the
createSelector
instance used by RTK Query - reworks slice selector logic to avoid depending on
this
value - fixes the order and inference of
create.asyncThunk
type parameters - fixes requirements for meta fields returned from
queryFn
s - marks promises that will never reject as safe, in preparation for https://github.com/typescript-eslint/typescript-eslint/issues/7008
What's Changed
- Update docs to avoid circular type by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3948
- Copy "Migrating to Modern Redux" and "RTK is Redux" docs from core site by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3956
- Fix store path rtk-query pokemon api tutorial by @suspiciousRaccoon in https://github.com/reduxjs/redux-toolkit/pull/3611
- tweak RTKQ without hooks section, and add note regarding memoization by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3963
- Add section regarding overriding deps by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3968
- Add section re: RR custom context typing change by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3995
- Add Expo demo app to CI workflow by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/3985
- docs: fix a typo in queries markdown file by @fatihgnc in https://github.com/reduxjs/redux-toolkit/pull/4013
- Preserve nullable store state type by avoiding intersection with {} by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4011
- Upgrade version of "graphql-request" by @phryneas in https://github.com/reduxjs/redux-toolkit/pull/4026
- [graphql-request-base-query] update RTK peerDependency by @phryneas in https://github.com/reduxjs/redux-toolkit/pull/4027
- fix: improve selectFromResult memoization by @thisjeremiah in https://github.com/reduxjs/redux-toolkit/pull/4029
- Rename "createSliceWithThunks" and "createThunkSlice" to "createAppSlice" by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4035
- Bump Vitest to latest version by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4047
- fix inference and order of async thunk generics by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4061
- Fix markdown links on Next.js page by @DmitryScaletta in https://github.com/reduxjs/redux-toolkit/pull/4069
- Introduce pre-typed listener middleware via
listenerMiddleware.withTypes<RootState, AppDispatch>()
method by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4049 - Add
.withTypes
tocreateDraftSafeSelector
by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4080 - Task/remove all settled by @bever1337 in https://github.com/reduxjs/redux-toolkit/pull/3917
- Allow customising createSelector instance used by RTKQ by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4048
- cacheLifecycle fix for fixedCacheKey in mutationThunk by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4082
- avoid relying on
this
in createSlice by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4071 - documentation: fix grammar in redux and redux toolkit comparison by @untilhamza in https://github.com/reduxjs/redux-toolkit/pull/4086
- Update
tsconfig.typetests.json
to include all TS files by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4091 - Remove abort event listner for AbortController by @Fonger in https://github.com/reduxjs/redux-toolkit/pull/3951
- Docs: unused value 'isRejected' in code snippet by @alphonsotran in https://github.com/reduxjs/redux-toolkit/pull/3301
- fix/kitchen-sink-isAuthenticated: fixed isAuthenticated state change on login fullfilled by @shrijan00003 in https://github.com/reduxjs/redux-toolkit/pull/3588
- Fix
composeWithDevTools
spy by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4093 - require queryFn meta to match base query by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/4098
- Added 'SafePromise' branded Promises for createAsyncThunk by @JoshuaKGoldberg in https://github.com/reduxjs/redux-toolkit/pull/4102
- Add React Native demo app to CI workflow by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/3984
- Modernize unit test setup by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/4114
- Add missing reducer field to ConfigureStoreOptions docs by @nickgirardo in https://github.com/reduxjs/redux-toolkit/pull/4116
- Option for queries to pause polling when unfocused by @riqts in https://github.com/reduxjs/redux-toolkit/pull/4055
New Contributors
- @suspiciousRaccoon made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/3611
- @fatihgnc made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4013
- @thisjeremiah made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4029
- @riqts made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4082
- @untilhamza made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4086
- @Fonger made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/3951
- @alphonsotran made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/3301
- @shrijan00003 made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/3588
- @JoshuaKGoldberg made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4102
- @nickgirardo made their first contribution in https://github.com/reduxjs/redux-toolkit/pull/4116
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v2.0.1...v2.1.0
v2.0.1
v2.0.0
This major release :
- Removes the deprecated object syntax from
createSlice
andcreateReducer
- Removes other deprecated options
- Updates the
middleware
andenhancers
options ofconfigureStore
to require callbacks - Updates the packaging for better ESM/CJS compatibility and modernizes the build output
- Includes all changes to Redux core 5.0, Reselect 5.0, and Redux Thunk 3.0
- Updates RTKQ default subscription behavior
- Adds a new
combineSlices
method with support for lazy-loading slice reducers - Adds a new "dynamic middleware" middleware with support for adding middleware at runtime
- Adds a new callback syntax to
createSlice.reducers
, with optional support for defining thunks inside ofcreateSlice
- Adds the
autoBatchEnhancer
toconfigureStore
by default - Has many additional TS tweaks and improvements
This release has breaking changes. (Note: v2.0.1 was released with a couple hotfixes for Reselect and Redux Thunk right as this was being finalized.)
This release is part of a wave of major versions of all the Redux packages: Redux Toolkit 2.0, Redux core 5.0, React-Redux 9.0, Reselect 5.0, and Redux Thunk 3.0.
For full details on all of the breaking changes and other significant changes to all of those packages, see the "Migrating to RTK 2.0 and Redux 5.0" migration guide in the Redux docs.
[!NOTE] The Redux core, Reselect, and Redux Thunk packages are included as part of Redux Toolkit, and RTK users do not need to manually upgrade them - you'll get them as part of the upgrade to RTK 2.0. (If you're not using Redux Toolkit yet, please start migrating your existing legacy Redux code to use Redux Toolkit today!)
### RTK
npm install @​reduxjs/toolkit
yarn add @​reduxjs/toolkit
Changelog
createSlice.extraReducers
and createReducer
removed
Object syntax for RTK's createReducer
API was originally designed to accept a lookup table of action type strings to case reducers, like { "ADD_TODO": (state, action) => {} }
. We later added the "builder callback" form to allow more flexibility in adding "matchers" and a default handler, and did the same for createSlice.extraReducers
.
We have removed the "object" form for both createReducer
and createSlice.extraReducers
in RTK 2.0, as the builder callback form is effectively the same number of lines of code, and works much better with TypeScript.
As an example, this:
const todoAdded = createAction('todos/todoAdded')
createReducer(initialState, {
[todoAdded]: (state, action) => {},
})
createSlice({
name,
initialState,
reducers: {
/* case reducers here */
},
extraReducers: {
[todoAdded]: (state, action) => {},
},
})
should be migrated to:
createReducer(initialState, (builder) => {
builder.addCase(todoAdded, (state, action) => {})
})
createSlice({
name,
initialState,
reducers: {
/* case reducers here */
},
extraReducers: (builder) => {
builder.addCase(todoAdded, (state, action) => {})
},
})
Codemods
To simplify upgrading codebases, we've published a set of codemods that will automatically transform the deprecated "object" syntax into the equivalent "builder" syntax.
The codemods package is available on NPM as @reduxjs/rtk-codemods
. More details are available here.
To run the codemods against your codebase, run npx @​reduxjs/rtk-codemods <TRANSFORM NAME> path/of/files/ or/some**/*glob.js.
Examples:
npx @​reduxjs/rtk-codemods createReducerBuilder ./src
npx @​reduxjs/rtk-codemods createSliceBuilder ./packages/my-app/**/*.ts
We also recommend re-running Prettier on the codebase before committing the changes.
These codemods should work, but we would greatly appreciate feedback from more real-world codebases!
configureStore
Options Changes
configureStore.middleware
must be a callback
Since the beginning, configureStore
has accepted a direct array value as the middleware
option. However, providing an array directly prevents configureStore
from calling getDefaultMiddleware()
. So, middleware: [myMiddleware]
means there is no thunk middleware added (or any of the dev-mode checks).
This is a footgun, and we've had numerous users accidentally do this and cause their apps to fail because the default middleware never got configured.
As a result, we've now made the middleware
only accept the callback form. If for some reason you still want to replace all of the built-in middleware, do so by returning an array from the callback:
const store = configureStore({
reducer,
middleware: (getDefaultMiddleware) => {
// WARNING: this means that _none_ of the default middleware are added!
return [myMiddleware]
// or for TS users, use:
// return new Tuple(myMiddleware)
},
})
But note that we consistently recommend not replacing the default middleware entirely, and that you should use return getDefaultMiddleware().concat(myMiddleware)
.
configureStore.enhancers
must be a callback
Similarly to configureStore.middleware
, the enhancers
field must also be a callback, for the same reasons.
The callback will receive a getDefaultEnhancers
function that can be used to customise the batching enhancer that's now included by default.
For example:
const store = configureStore({
reducer,
enhancers: (getDefaultEnhancers) => {
return getDefaultEnhancers({
autoBatch: { type: 'tick' },
}).concat(myEnhancer)
},
})
It's important to note that the result of getDefaultEnhancers
will also contain the middleware enhancer created with any configured/default middleware. To help prevent mistakes, configureStore
will log an error to console if middleware was provided and the middleware enhancer wasn't included in the callback result.
const store = configureStore({
reducer,
enhancers: (getDefaultEnhancers) => {
return [myEnhancer] // we've lost the middleware here
// instead:
return getDefaultEnhancers().concat(myEnhancer)
},
})
Also, note that if you supply the enhancers
field, it must come after the middleware
field in order for TS inference to work properly.
getDefaultMiddleware
and getType
removed
Standalone The standalone version of getDefaultMiddleware
has been deprecated since v1.6.1, and has now been removed. Use the function passed to the middleware
callback instead, which has the correct types.
We have also removed the getType
export, which was used to extract a type string from action creators made with createAction
. Instead, use the static property actionCreator.type
.
RTK Query behaviour changes
We've had a number of reports where RTK Query had issues around usage of dispatch(endpoint.initiate(arg, {subscription: false}))
. There were also reports that multiple triggered lazy queries were resolving the promises at the wrong time. Both of these had the same underlying issue, which was that RTKQ wasn't tracking cache entries in these cases (intentionally). We've reworked the logic to always track cache entries (and remove them as needed), which should resolve those behavior issues.
We also have had issues raised about trying to run multiple mutations in a row and how tag invalidation behaves. RTKQ now has internal logic to delay tag invalidation briefly, to allow multiple invalidations to get handled together. This is controlled by a new invalidationBehavior: 'immediate' | 'delayed'
flag on createApi
. The new default behavior is 'delayed'
. Set it to 'immediate'
to revert to the behavior in RTK 1.9.
In RTK 1.9, we reworked RTK Query's internals to keep most of the subscription status inside the RTKQ middleware. The values are still synced to the Redux store state, but this is primarily for display by the Redux DevTools "RTK Query" panel. Related to the cache entry changes above, we've optimized how often those values get synced to the Redux state for perf.
ESM/CJS Package Compatibility
The biggest theme of the Redux v5 and RTK 2.0 releases is trying to get "true" ESM package publishing compatibility in place, while still supporting CJS in the published package.
The primary build artifact is now an ESM file, dist/redux-toolkit.modern.mjs
. Most build tools should pick this up. There's also a CJS artifact, and a second copy of the ESM file named redux-toolkit.legacy-esm.js
to support Webpack 4 (which does not recognize the exports
field in package.json
). Additionally, all of the build artifacts now live under ./dist/
in the published package.
Modernized Build Output
We now publish modern JS syntax targeting ES2020, including optional chaining, object spread, and other modern syntax. If you need to target older browsers, please transpile the packages yourself (or use the legacy-esm
build artifact for ES2017).
Build Tooling
We're now building the package using https://github.com/egoist/tsup. We also now include sourcemaps for the ESM and CJS artifacts.
Dropping UMD Builds
Redux has always shipped with UMD build artifacts. These are primarily meant for direct import as script tags, such as in a CodePen or a no-bundler build environment.
We've dropped those build artifacts from the published package, on the grounds that the use cases seem pretty rare today.
There's now a redux-toolkit.browser.mjs
file in the package that can be loaded from a CDN like Unpkg.
If you have strong use cases for us continuing to include UMD build artifacts, please let us know!
Dependency Updates
Redux Libraries
RTK now depends on Redux core 5.0, Reselect 5.0, and Redux Thunk 3.0. See the linked release notes for those libraries, as each of them has additional breaking changes. The "Migrating to RTK 2.0 and Redux 5.0" docs page also covers the combined changes in one page
Immer 10
RTK now also depends on Immer 10.0, which has several major improvements and updates:
- Much faster update perf
- Much smaller bundle size
- Better ESM/CJS package formatting
- No default export
- No ES5 fallback
We've also removed the prior call to automatically enable the Immer ES5 fallback mode any time RTK was loaded.
Other Changes
Bundle Size Optimizations
Redux 4.1.0 optimized its bundle size by extracting error message strings out of production builds, based on React's approach. We've applied the same technique to RTK. This saves about 1000 bytes from prod bundles (actual benefits will depend on which imports are being used).
We also noted that ESBuild does not deduplicate imports when it bundles source files, and this was causing RTK Query's bundle to contain a dozen references to import { } from "@​reduxjs/toolkit"
, including some of the same methods. Manually deduplicating those saves about 600 bytes off the production RTKQ artifact.
reactHooksModule
custom hook configuration
Previously, custom versions of React Redux's hooks (useSelector
, useDispatch
, and useStore
) could be passed separately to reactHooksModule
, usually to enable using a different context to the default ReactReduxContext
.
In practicality, the react hooks module needs all three of these hooks to be provided, and it became an easy mistake to only pass useSelector
and useDispatch
, without useStore
.
The module has now moved all three of these under the same configuration key, and will check that all three are provided if the key is present.
// previously
const customCreateApi = buildCreateApi(
coreModule(),
reactHooksModule({
useDispatch: createDispatchHook(MyContext),
useSelector: createSelectorHook(MyContext),
useStore: createStoreHook(MyContext),
})
)
// now
const customCreateApi = buildCreateApi(
coreModule(),
reactHooksModule({
hooks: {
useDispatch: createDispatchHook(MyContext),
useSelector: createSelectorHook(MyContext),
useStore: createStoreHook(MyContext),
},
})
)
Deprecated Options Removed
Several other options were previously marked as deprecated, and have been removed. We've also removed polyfills like the AbortController
polyfill.
TypeScript Changes
configureStore
field order for middleware
matters
If you are passing both the middleware
and enhancers
fields to configureStore
, the middleware
field must come first in order for internal TS inference to work properly.
Tuple
Non-default middleware/enhancers must use We've seen many cases where users passing the middleware
parameter to configureStore have tried spreading the array returned by getDefaultMiddleware()
, or passed an alternate plain array. This unfortunately loses the exact TS types from the individual middleware, and often causes TS problems down the road (such as dispatch
being typed as Dispatch<AnyAction>
and not knowing about thunks).
getDefaultMiddleware()
already used an internal MiddlewareArray
class, an Array
subclass that had strongly typed .concat/prepend()
methods to correctly capture and retain the middleware types.
We've renamed that type to Tuple
, and configureStore
's TS types now require that you must use Tuple
if you want to pass your own array of middleware:
import { configureStore, Tuple } from '@​reduxjs/toolkit'
configureStore({
reducer: rootReducer,
middleware: (getDefaultMiddleware) => new Tuple(additionalMiddleware, logger),
})
(Note that this has no effect if you're using RTK with plain JS, and you could still pass a plain array here.)
This same restriction applies to the enhancers
field.
Entity adapter type updates
createEntityAdapter
now has an Id
generic argument, which will be used to strongly type the item IDs anywhere those are exposed. Previously, the ID field type was always string | number
. TS will now try to infer the exact type from either the .id
field of your entity type, or the selectId
return type. You could also fall back to passing that generic type directly. If you use the EntityState<Data, Id>
type directly, you must supply both generic arguments!
The .entities
lookup table is now defined to use a standard TS Record<Id, MyEntityType>
, which assumes that each item lookup exists by default. Previously, it used a Dictionary<MyEntityType>
type, which assumed the result was MyEntityType | undefined
. The Dictionary
type has been removed.
If you prefer to assume that the lookups might be undefined, use TypeScript's noUncheckedIndexedAccess
configuration option to control that.
New Features
These features are new in Redux Toolkit 2.0, and help cover additional use cases that we've seen users ask for in the ecosystem.
combineSlices
API with slice reducer injection for code-splitting
The Redux core has always included combineReducers
, which takes an object full of "slice reducer" functions and generates a reducer that calls those slice reducers. RTK's createSlice
generates slice reducers + associated action creators, and we've taught the pattern of exporting individual action creators as named exports and the slice reducer as a default export. Meanwhile, we've never had official support for lazy-loading reducers, although we've had sample code for some "reducer injection" patterns in our docs.
This release includes a new combineSlices
API that is designed to enable lazy-loading of reducers at runtime. It accepts individual slices or an object full of slices as arguments, and automatically calls combineReducers
using the sliceObject.name
field as the key for each state field. The generated reducer function has an additional .inject()
method attached that can be used to dynamically inject additional slices at runtime. It also includes a .withLazyLoadedSlices()
method that can be used to generate TS types for reducers that will be added later. See #2776 for the original discussion around this idea.
For now, we are not building this into configureStore
, so you'll need to call const rootReducer = combineSlices(.....)
yourself and pass that to configureStore({reducer: rootReducer})
.
Basic usage: a mixture of slices and standalone reducers passed to combineSlices
const stringSlice = createSlice({
name: 'string',
initialState: '',
reducers: {},
})
const numberSlice = createSlice({
name: 'number',
initialState: 0,
reducers: {},
})
const booleanReducer = createReducer(false, () => {})
const api = createApi(/* */)
const combinedReducer = combineSlices(
stringSlice,
{
num: numberSlice.reducer,
boolean: booleanReducer,
},
api
)
expect(combinedReducer(undefined, dummyAction())).toEqual({
string: stringSlice.getInitialState(),
num: numberSlice.getInitialState(),
boolean: booleanReducer.getInitialState(),
api: api.reducer.getInitialState(),
})
Basic slice reducer injection
// Create a reducer with a TS type that knows `numberSlice` will be injected
const combinedReducer =
combineSlices(stringSlice).withLazyLoadedSlices<
WithSlice<typeof numberSlice>
>()
// `state.number` doesn't exist initially
expect(combinedReducer(undefined, dummyAction()).number).toBe(undefined)
// Create a version of the reducer with `numberSlice` injected (mainly useful for types)
const injectedReducer = combinedReducer.inject(numberSlice)
// `state.number` now exists, and injectedReducer's type no longer marks it as optional
expect(injectedReducer(undefined, dummyAction()).number).toBe(
numberSlice.getInitialState()
)
// original reducer has also been changed (type is still optional)
expect(combinedReducer(undefined, dummyAction()).number).toBe(
numberSlice.getInitialState()
)
selectors
field in createSlice
The existing createSlice
API now has support for defining selectors
directly as part of the slice. By default, these will be generated with the assumption that the slice is mounted in the root state using slice.name
as the field, such as name: "todos"
-> rootState.todos
. Additionally, there's now a slice.selectSlice
method that does that default root state lookup.
You can call sliceObject.getSelectors(selectSliceState)
to generate the selectors with an alternate location, similar to how entityAdapter.getSelectors()
works.
const slice = createSlice({
name: 'counter',
initialState: 42,
reducers: {},
selectors: {
selectSlice: (state) => state,
selectMultiple: (state, multiplier: number) => state * multiplier,
},
})
// Basic usage
const testState = {
[slice.name]: slice.getInitialState(),
}
const { selectSlice, selectMultiple } = slice.selectors
expect(selectSlice(testState)).toBe(slice.getInitialState())
expect(selectMultiple(testState, 2)).toBe(slice.getInitialState() * 2)
// Usage with the slice reducer mounted under a different key
const customState = {
number: slice.getInitialState(),
}
const { selectSlice, selectMultiple } = slice.getSelectors(
(state: typeof customState) => state.number
)
expect(selectSlice(customState)).toBe(slice.getInitialState())
expect(selectMultiple(customState, 2)).toBe(slice.getInitialState() * 2)
createSlice.reducers
callback syntax and thunk support
One of the oldest feature requests we've had is the ability to declare thunks directly inside of createSlice
. Until now, you've always had to declare them separately, give the thunk a string action prefix, and handle the actions via createSlice.extraReducers
:
// Declare the thunk separately
const fetchUserById = createAsyncThunk(
'users/fetchByIdStatus',
async (userId: number, thunkAPI) => {
const response = await userAPI.fetchById(userId)
return response.data
}
)
const usersSlice = createSlice({
name: 'users',
initialState,
reducers: {
// standard reducer logic, with auto-generated action types per reducer
},
extraReducers: (builder) => {
// Add reducers for additional action types here, and handle loading state as needed
builder.addCase(fetchUserById.fulfilled, (state, action) => {
state.entities.push(action.payload)
})
},
})
Many users have told us that this separation feels awkward.
We've wanted to include a way to define thunks directly inside of createSlice
, and have played around with various prototypes. There were always two major blocking issues, and a secondary concern:
- It wasn't clear what the syntax for declaring a thunk inside should look like.
- Thunks have access to
getState
anddispatch
, but theRootState
andAppDispatch
types are normally inferred from the store, which in turn infers it from the slice state types. Declaring thunks insidecreateSlice
would cause circular type inference errors, as the store needs the slice types but the slice needs the store types. We weren't willing to ship an API that would work okay for our JS users but not for our TS users, especially since we want people to use TS with RTK. - You can't do synchronous conditional imports in ES modules, and there's no good way to make the
createAsyncThunk
import optional. EithercreateSlice
always depends on it (and adds that to the bundle size), or it can't usecreateAsyncThunk
at all.
We've settled on these compromises:
-
In order to create async thunks with
createSlice
, you specifically need to set up a custom version ofcreateSlice
that has access tocreateAsyncThunk
. - You can declare thunks inside of
createSlice.reducers
, by using a "creator callback" syntax for thereducers
field that is similar to thebuild
callback syntax in RTK Query'screateApi
(using typed functions to create fields in an object). Doing this does look a bit different than the existing "object" syntax for thereducers
field, but is still fairly similar. - You can customize some of the types for thunks inside of
createSlice
, but you cannot customize thestate
ordispatch
types. If those are needed, you can manually do anas
cast, likegetState() as RootState
.
In practice, we hope these are reasonable tradeoffs. Creating thunks inside of createSlice
has been widely asked for, so we think it's an API that will see usage. If the TS customization options are a limitation, you can still declare thunks outside of createSlice
as always, and most async thunks don't need dispatch
or getState
- they just fetch data and return. And finally, setting up a custom createSlice
allows you to opt into createAsyncThunk
being included in your bundle size (though it may already be included if used directly or as part of RTK Query - in either of these cases there's no additional bundle size).
Here's what the new callback syntax looks like:
const createSliceWithThunks = buildCreateSlice({
creators: { asyncThunk: asyncThunkCreator },
})
const todosSlice = createSliceWithThunks({
name: 'todos',
initialState: {
loading: false,
todos: [],
error: null,
} as TodoState,
reducers: (create) => ({
// A normal "case reducer", same as always
deleteTodo: create.reducer((state, action: PayloadAction<number>) => {
state.todos.splice(action.payload, 1)
}),
// A case reducer with a "prepare callback" to customize the action
addTodo: create.preparedReducer(
(text: string) => {
const id = nanoid()
return { payload: { id, text } }
},
// action type is inferred from prepare callback
(state, action) => {
state.todos.push(action.payload)
}
),
// An async thunk
fetchTodo: create.asyncThunk(
// Async payload function as the first argument
async (id: string, thunkApi) => {
const res = await fetch(`myApi/todos?id=${id}`)
return (await res.json()) as Item
},
// An object containing `{pending?, rejected?, fulfilled?, settled?, options?}` second
{
pending: (state) => {
state.loading = true
},
rejected: (state, action) => {
state.error = action.payload ?? action.error
},
fulfilled: (state, action) => {
state.todos.push(action.payload)
},
// settled is called for both rejected and fulfilled actions
settled: (state, action) => {
state.loading = false
},
}
),
}),
})
// `addTodo` and `deleteTodo` are normal action creators.
// `fetchTodo` is the async thunk
export const { addTodo, deleteTodo, fetchTodo } = todosSlice.actions
Codemod
Using the new callback syntax is entirely optional (the object syntax is still standard), but an existing slice would need to be converted before it can take advantage of the new capabilities this syntax provides. To make this easier, a codemod is provided.
npx @​reduxjs/rtk-codemods createSliceReducerBuilder ./src/features/todos/slice.ts
"Dynamic middleware" middleware
A Redux store's middleware pipeline is fixed at store creation time and can't be changed later. We have seen ecosystem libraries that tried to allow dynamically adding and removing middleware, potentially useful for things like code splitting.
This is a relatively niche use case, but we've built our own version of a "dynamic middleware" middleware. Add it to the Redux store at setup time, and it lets you add middleware later at runtime. It also comes with a React hook integration that will automatically add a middleware to the store and return the updated dispatch method..
import { createDynamicMiddleware, configureStore } from '@​reduxjs/toolkit'
const dynamicMiddleware = createDynamicMiddleware()
const store = configureStore({
reducer: {
todos: todosReducer,
},
middleware: (getDefaultMiddleware) =>
getDefaultMiddleware().prepend(dynamicMiddleware.middleware),
})
// later
dynamicMiddleware.addMiddleware(someOtherMiddleware)
configureStore
adds autoBatchEnhancer
by default
In v1.9.0, we added a new autoBatchEnhancer
that delays notifying subscribers briefly when multiple "low-priority" actions are dispatched in a row. This improves perf, as UI updates are typically the most expensive part of the update process. RTK Query marks most of its own internal actions as "low-pri" by default, but you have to have the autoBatchEnhancer
added to the store to benefit from that.
We've updated configureStore
to add the autoBatchEnhancer
to the store setup by default, so that users can benefit from the improved perf without needing to manually tweak the store config themselves.
entityAdapter.getSelectors
accepts a createSelector
function
entityAdapter.getSelectors()
now accepts an options object as its second argument. This allows you to pass in your own preferred createSelector
method, which will be used to memoize the generated selectors. This could be useful if you want to use one of Reselect's new alternate memoizers, or some other memoization library with an equivalent signature.
Next.js Setup Guide
We now have a docs page that covers how to set up Redux properly with Next.js. We've seen a lot of questions around using Redux, Next, and the App Router together, and this guide should help provide advice.
(At this time, the Next.js with-redux
example is still showing outdated patterns - we're going to file a MR shortly to update that to match our docs guide.)
What's Changed
- Remove legacy object syntax for reducers by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3051
- Upgrade build tooling by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3088
- Migrate the RTK package to be full ESM by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3095
- Migrate RTK test suite from Jest to Vitest by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3102
- Fix type errors after upgrading to Redux 5 alpha by @Methuselah96 in https://github.com/reduxjs/redux-toolkit/pull/3177
- Bump Redux dep to 5.0.0-alpha.2 by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3170
- Test published artifacts in CI by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3213
- Merge RTK CI examples into
v2.0-integration
by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3253 - Add
arethetypeswrong
automated CLI check by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3294 - update tip regarding overrideExisting to match actual behaviour by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3305
- Add
attw
CLI option to treat problems as non-errors by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3316 - Rework build setup and hopefully fix ESM compat issues by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3318
- Bump Immer to 10.0-beta by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3320
- Switch build setup from a custom ESBuild+TS script to
tsup
by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3362 - Use original instead of immer draft for perf by @GeorchW in https://github.com/reduxjs/redux-toolkit/pull/3270
- enable enhanceEndpoints.transformResponse to override ResultType by @dmitrigrabov in https://github.com/reduxjs/redux-toolkit/pull/2953
- Fix global
responseHandler
being used infetchBaseQuery
by @praxxis in https://github.com/reduxjs/redux-toolkit/pull/3137 - reset internalState.currentSubscriptions on
resetApiState
by @phryneas in https://github.com/reduxjs/redux-toolkit/pull/3333 - Bump deps and mark
subscriptionUpdated
as autobatched by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3364 - Redux 5alpha5 by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3367
- add isAction helper function, and ensure listener middleware only runs for actions by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3372
- Allow inference of enhancer state extensions, and fix inference when using callback form by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3207
- combineSlices implementation by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3297
- Bump Immer to 10.0 final by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3376
- Allow partial preloaded state for combined slice reducer and update devDeps by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3381
- ensure it's only possible to pass all or none of the hooks to reactHooksModule by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3400
- Remove remaining deprecated exports by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3398
- create action creator middleware by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3414
- Restore query status types by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3420
- Implement auto fork joining by @ericanderson in https://github.com/reduxjs/redux-toolkit/pull/3407
- types: make it easier to wrap createAsyncThunk by @shrouxm in https://github.com/reduxjs/redux-toolkit/pull/3393
- Fixed Stackoverflow bug if children prop is a ref to root/parent object by @cheprasov in https://github.com/reduxjs/redux-toolkit/pull/3428
- Add getDefaultEnhancers callback, and add autoBatchEnhancer to defaults. by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3347
- Update to Redux alpha 6 by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3442
- Remove toString override from action creators, in favour of explicit .type field. by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3425
- Add "creator callback" syntax to slice reducer field, to allow for async thunk creation by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3388
- API for adding middleware dynamically by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3292
- Add Id type parameter in createEntityAdapter by @Matt-Ord in https://github.com/reduxjs/redux-toolkit/pull/3187
- Require usage of Tuple in TS by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3460
- Require that enhancers is a callback by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3461
- fix invalid createEntityAdapter call by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3490
- Use Record<EntityId, T> instead of Dictionary by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3424
- Fix inference for reducers without actions specified by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3475
- Prefer UnknownAction and Action to AnyAction by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3363
- Allow passing a
createSelector
instance to adapter.getSelectors by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3481 - add required type parameter to EntityState in docs by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3496
- fix syntax for users with noPropertyAccessFromIndexSignature enabled by @evertbouw in https://github.com/reduxjs/redux-toolkit/pull/3495
- Fix TransformedResponse type to unwrap promise by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3500
- Fix multiple TS build and packaging issues by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3672
- fix: Default export of the module has or is using private name type error when using latest alpha/beta by @eric-crowell in https://github.com/reduxjs/redux-toolkit/pull/3605
- RTK Query: resolve type portability issues by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3678
- fix(RTKQuery React): Resolved declare module pathing for Node16 by @eric-crowell in https://github.com/reduxjs/redux-toolkit/pull/3592
- fix(2448): remove abort controller polyfill by @JulienKode in https://github.com/reduxjs/redux-toolkit/pull/2518
- Throw error when type is empty in builder.addCase by @chawes13 in https://github.com/reduxjs/redux-toolkit/pull/3572
- [RED-23] fix: Updated type references to resolve portable types issue by @tdurnford in https://github.com/reduxjs/redux-toolkit/pull/3728
- add option to update provided tags by @dutzi in https://github.com/reduxjs/redux-toolkit/pull/3255
- Add selectCachedArgsForQuery util by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3547
- [RED-26] Remove Request.clone() usage in fetchBaseQuery by @alex-vukov in https://github.com/reduxjs/redux-toolkit/pull/3720
- Try working around TS 4.1 mismatch by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3739
- Optimize bundles for RTK 2.0 by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3740
- Work around known TS bug with type inference by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3761
- Remove middleware array option by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3760
- warn for and migrate old options and require all hooks by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3549
- [RED-24] Feature/entity adapter draftable entity state by @Olesj-Bilous in https://github.com/reduxjs/redux-toolkit/pull/3669
- write create callback codemod by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3525
- Copy of "Work around known TS bug with type inference #3761" by @julian-ford in https://github.com/reduxjs/redux-toolkit/pull/3777
- Rework named hooks type (v1.9) by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3769
- Rewrite HooksWithUniqueNames type (v2.0) by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3767
- Adds the option of generating real TS enums instead of string unions by @labmorales in https://github.com/reduxjs/redux-toolkit/pull/2854
- fix: adjust snake case regex by @juliengbt in https://github.com/reduxjs/redux-toolkit/pull/3785
- Update error message: name of action type by @Kalibryyy in https://github.com/reduxjs/redux-toolkit/pull/3800
- Created listenerApi.cancel() by @julian-ford in https://github.com/reduxjs/redux-toolkit/pull/3775
- Add
listenerApi.throwIfCancelled()
by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3802 - Add settled matcher for createAsyncThunk by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3768
- Rewrite RTKQ internal subscription lookups and subscription syncing by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3824
- keep subscription on data while query is running by @phryneas in https://github.com/reduxjs/redux-toolkit/pull/3709
- Stop re-exporting
autotrackMemoize
by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3831 - Delayed tag invalidations by @GeorchW in https://github.com/reduxjs/redux-toolkit/pull/3116
- [RTK v2.0] output selector fields are currently missing in selector functions created using
createDraftSafeSelector
. by @aryaemami59 in https://github.com/reduxjs/redux-toolkit/pull/3722 - createDynamicMiddleware bike shedding by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3763
- Allow passing selector creators with different memoize options to getSelectors by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3833
- Add selectSlice to slice instance by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3838
- Throw an error if ApiProvider is nested inside a normal Provider. by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3855
- Update store configuration using configureStore in createSlice.mdx by @PaoloDiBello in https://github.com/reduxjs/redux-toolkit/pull/3869
- Create standardised methods of modifying reducer handler context. by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3872
- Require calling buildCreateSlice to use create.asyncThunk by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3867
- Restore the toString override, but keep it out of the docs. by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3877
- Selector housekeeping - emplace and unwrapped by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3878
- Update build tooling for 2.0 by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3902
- Export EndpointBuilder by @golergka in https://github.com/reduxjs/redux-toolkit/pull/3893
- Update to Redux core v5 rc1 and replace isAction/isPlainObject with core counterparts by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3904
- Drop "ESM modern dev" build and rename "ESM prod" to "browser" by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3905
- fix isFSA import and avoid unnecessary type param by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3906
- Avoid passing an identity function to createSelector by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3931
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v1.9.3...v2.0.0
v1.9.7
This bugfix release rewrites the RTKQ hook TS types to significantly improve TS perf.
Changelog
RTKQ TS Perf
A number of users had reported that Intellisense for RTKQ API objects was extremely slow (multiple seconds) - see discussion in #3214 . We did some perf investigation on user-provided examples, and concluded that the biggest factor to slow RTKQ TS perf was the calculation of hook names like useGetPokemonQuery
, which was generating a large TS union of types.
We've rewritten that hook names type calculation to use mapped types and a couple of intersections. In a specific user-provided stress test repo, it dropped TS calculation time by 60% (2600ms to 1000ms).
There's more potential work we can do to improve things, but this seems like a major perf improvement worth shipping now.
What's Changed
- chore: Switch 4.9.2-rc to 4.9.5 since 4.9.5 has been released in TypeScript by @kahirokunn in https://github.com/reduxjs/redux-toolkit/pull/3772
- Copy of "Work around known TS bug with type inference #3761" by @julian-ford in https://github.com/reduxjs/redux-toolkit/pull/3777
- Rework named hooks type (v1.9) by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3769
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v1.9.6...v1.9.7
v1.9.6
This bugfix release adds a new dev-mode middleware to catch accidentally dispatching an action creator, adds a new listener middleware option around waiting for forks, adds a new option to update provided tags when updateQueryData
is used, reworks internal types to better handle uses with TS declaration output, and fixes a variety of small issues.
Changelog
Action Creator Dev Check Middleware
RTK already includes dev-mode middleware that check for the common mistakes of accidentally mutating state and putting non-serializable values into state or actions.
Over the years we've also seen a semi-frequent error where users accidentally pass an action creator reference to dispatch
, instead of calling it and dispatching the action it returns.
We've added another dev-mode middleware that specifically catches this error and warns about it.
Additional Options
The listener middleware's listenerApi.fork()
method now has an optional autoJoin
flag that can be used to keep the effect from finishing until all active forked tasks have completed.
updateQueryData
now has an updateProvidedTags
option that will force a recalculation of that endpoint's provided tags. It currently defaults to false
, and we'll likely turn that to true
in the next major.
Other Fixes
The builder.addCase
method now throws an error if a type
string is empty.
fetchBaseQuery
now uses an alternate method to clone the original Request
in order to work around an obscure Chrome bug.
The immutability middleware logic was tweaked to avoid a potential stack overflow.
Types Changes
The internal type imports have been reworked to try to fix "type portability" issues when used in combination with TS declaration outputs.
A couple additional types were exported to help with wrapping createAsyncThunk
.
What's Changed
- create action creator middleware by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3414
- Implement auto fork joining by @ericanderson in https://github.com/reduxjs/redux-toolkit/pull/3407
- types: make it easier to wrap createAsyncThunk by @shrouxm in https://github.com/reduxjs/redux-toolkit/pull/3393
- Fixed Stackoverflow bug if children prop is a ref to root/parent object by @cheprasov in https://github.com/reduxjs/redux-toolkit/pull/3428
- Fix TransformedResponse type to unwrap promise by @EskiMojo14 in https://github.com/reduxjs/redux-toolkit/pull/3500
- Throw error when type is empty in builder.addCase by @chawes13 in https://github.com/reduxjs/redux-toolkit/pull/3572
- [RED-23] fix: Updated type references to resolve portable types issue by @tdurnford in https://github.com/reduxjs/redux-toolkit/pull/3728
- add option to update provided tags by @dutzi in https://github.com/reduxjs/redux-toolkit/pull/3255
- [RED-26] Remove Request.clone() usage in fetchBaseQuery by @alex-vukov in https://github.com/reduxjs/redux-toolkit/pull/3720
- Try working around TS 4.1 mismatch by @markerikson in https://github.com/reduxjs/redux-toolkit/pull/3739
Full Changelog: https://github.com/reduxjs/redux-toolkit/compare/v1.9.5...v1.9.6
Configuration
-
If you want to rebase/retry this MR, click this checkbox.
This MR has been generated by Renovate Bot.