Releases: reduxjs/redux-toolkit
RTK-Query OpenAPI Codegen v1.1.0
Changelog
Added:
- Option of generating real TS enums instead of string unions Adds the option of generating real TS enums instead of string unions #2854
- Compatibility with TypeScript 5.x versions as the codegen relies on the TypeScript AST for code generation
- As a result also needs a higher TypeScript version to work with (old version range was 4.1-4.5)
- Changes dependency from a temporarily patched old version of
oazapftsback to the current upstream version
v2.2.7
This bugfix release fixes issues with "TS type portability" errors, improves build artifact tree shaking behavior, and exports some additional TS types.
Changelog
TS Type Portability
We've had a slew of issues reported around "TS type portability" errors, such as:
The error messages are typically along the lines of:
Type error: The inferred type of
'configureStore'cannot be named without a reference to'@reduxjs/toolkit/node_modules/redux'. This is likely not portable. A type annotation is necessary.
@aryaemami59 did some deep investigation and concluded these were due to a mixture of using interface instead of type in most places, not pre-bundling our TS typedefs, and not exporting some of the unique symbols we use internally.
Arya put together a highly detailed writeup and set of fixes in #4467: Fix: TypeScript Type Portability Issues, and that appears to resolve all of those issues we've seen. Thank you!
Other Changes
Arya also did significant work to improve RTK's treeshaking, tweaking internal definitions to let bundlers better separate out unused code.
We've exported additional types like UpdateDefinitions and RetryOptions, per request.
listenerMiddleware.withTypes() methods now allow passing in an ExtraArgument generic.
What's Changed
- Fix: TypeScript Type Portability Issues by @aryaemami59 in #4467
- Export UpdateDefinitions from toolkit/query by @joekrill in #4519
- feat: Adds ExtraArgument to withTypes for listenerMiddleware. by @antondalgren in #4517
- Improve treeshakeability of build artifacts by @aryaemami59 in #4435
- Export RetryOptions by @markerikson in #4529
Full Changelog: v2.2.6...v2.2.7
v2.2.6
This bugfix release:
- Brings internal
useIsomorphicLayoutEffectusage in line with React Redux in React Native environments - Exports
FetchBaseQueryArgstype - Fixes an issue in recent
createEntityAdaptersorting perf improvements that could (in specific cases) cause Immer to throw an error
What's Changed
- Fix
useIsomorphicLayoutEffectusage in React Native environments by @aryaemami59 in #4436 - Add export for FetchBaseQueryArgs by @WilsonCWong in #4465
- Fix Immer current usage when calling addManyMutably more than once by @JacobShafer in #4420
- Golf a few pieces of entity adapter code by @EskiMojo14 in #4485
Full Changelog: v2.2.5...v2.2.6
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
currentmay fail if the value is not a draft by @markerikson in #4412
Full Changelog: 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
Contextreferences to match the new nullableContextby @aryaemami59 in #4336 - types: export
GetThunkAPItype by @shrouxm in #4289 - Inline the deprecated
React.ReactChildtype by @aryaemami59 in #4382 - Replace usage of deprecated
JSXglobal namespace withReact.JSXby @aryaemami59 in #4381 - Fix
useRefusages to be called with an explicit argument ofundefined. by @aryaemami59 in #4380 - Bump
reselectdependency to 5.1.0 to resolve #4200 by @aryaemami59 in #4400 - Improve sorted entity adapter sorting performance by @markerikson in #4361
- Prevent inference of any in create.asyncThunk by @EskiMojo14 in #4389
- try out simplifying matcher code by @EskiMojo14 in #4386
- Handle bigint in useQuerySubscription by @rkofman in #4315
- fix: ensure hook subscription failures do not reset isLoading state by @smacpherson64 in #4364
- Improve usability of the
initiate()action for mutations. by @jared-ca in #4337 - Make retryCondition receive unknown error, as the user may not be using fetchBaseQuery by @EskiMojo14 in #3957
Full Changelog: 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
.withTypesby @aryaemami59 in #4308
Full Changelog: 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 #4204
- Fixed memory leak in rapid hook arg changing by @riqts in #4268
- Fix incorrect legacy-esm target by @markerikson in #4284
Full Changelog: v2.2.0...v2.2.2
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
undefinedif no additional properties are desired
- Equivalent to
- Allows initialising
combineSliceswith 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 foroverrideExistingininjectEndpoints, 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.isPrefetchvalue to query thunk actions when prefetched
What's Changed
- Revamp type tests setup by @aryaemami59 in #4095
- Bump Prettier and Prettier related packages by @aryaemami59 in #4126
- Fix codemods to work with TypeScript 4.7+ by @aryaemami59 in #4081
- Export types related to overriding the result types by @aryaemami59 in #4134
- Migrate type tests to Vitest by @aryaemami59 in #4127
- Fix RetryOptions type test by flipping order by @EskiMojo14 in #4136
- Format all files by @aryaemami59 in #4135
- [Docs/Website] skipPollingIfUnfocused added to polling overview and query options by @riqts in #4131
- [Docs] Changed create.asyncThunk setup admonition to caution by @riqts in #4150
- Temporarily revert node-fetch bump until ESM issues are sorted by @EskiMojo14 in #4151
- Contributing Markdown Refresh by @itz-Me-Pj in #4139
- investigate re-adding size limit action by @EskiMojo14 in #4083
- Add size limit imports for more commonly used RTK exports by @EskiMojo14 in #4155
- pure some things by @EskiMojo14 in #4157
- cut back on suffixes tested for size by @EskiMojo14 in #4160
- Incorrect builder method referenced by @kantbtrue in #4161
- Rename
cli.jstocli.mjsby @aryaemami59 in #4169 - Migrate Codegen OpenAPI's unit tests to Vitest by @aryaemami59 in #4137
- Fix wrong state for injected slices when not declared via
withLazyLoadedSlicesby @aryaemami59 in #4172 - [Docs, createEntityAdapter API]: add missing setOne and setMany signature by @kyselberg in #4173
- Corrected the "Observing cache behaviour" example point no.4 by @721-atikshaikh in #4174
- feat: add isPrefetch property in query action by @juliengbt in #4177
- allow initialising combined slice reducer with no static slices by @EskiMojo14 in #4184
- Create more Typed wrappers for RTKQ hook types by @EskiMojo14 in #4147
- proposal fix for axios base query types by @smff in #4186
- Provide 'throw' option for
overrideExistingby @ffluk3 in #4189 - Use vite-tsconfig-paths to make path aliasing easier by @aryaemami59 in #4175
- Remove trailing commas in
tsconfig.jsonfiles of all CodesandBox examples by @aryaemami59 in #4190 - [Docs] Added withTypes documentation in createDraftSafeSelector by @riqts in #4143
- Add second parameter to getInitialState to prefill entities by @EskiMojo14 in #4183
New Contributors
- @itz-Me-Pj made their first contribution in #4139
- @kantbtrue made their first contribution in #4161
- @kyselberg made their first contribution in #4173
- @721-atikshaikh made their first contribution in #4174
- @smff made their first contribution in #4186
- @ffluk3 made their first contribution in #4189
Full Changelog: v2.1.0...v2.2.0
v2.1.0
This minor release:
- adds withTypes methods to
listenerMiddlewareandcreateDraftSafeSelector - adds a
skipPollingIfUnfocusedoption to RTK Query - adds the ability to customise the
createSelectorinstance used by RTK Query - reworks slice selector logic to avoid depending on
thisvalue - fixes the order and inference of
create.asyncThunktype parameters - fixes requirements for meta fields returned from
queryFns - marks promises that will never reject as safe, in preparation for typescript-eslint/typescript-eslint#7008
What's Changed
- Update docs to avoid circular type by @EskiMojo14 in #3948
- Copy "Migrating to Modern Redux" and "RTK is Redux" docs from core site by @markerikson in #3956
- Fix store path rtk-query pokemon api tutorial by @suspiciousRaccoon in #3611
- tweak RTKQ without hooks section, and add note regarding memoization by @EskiMojo14 in #3963
- Add section regarding overriding deps by @EskiMojo14 in #3968
- Add section re: RR custom context typing change by @EskiMojo14 in #3995
- Add Expo demo app to CI workflow by @aryaemami59 in #3985
- docs: fix a typo in queries markdown file by @fatihgnc in #4013
- Preserve nullable store state type by avoiding intersection with {} by @EskiMojo14 in #4011
- Upgrade version of "graphql-request" by @phryneas in #4026
- [graphql-request-base-query] update RTK peerDependency by @phryneas in #4027
- fix: improve selectFromResult memoization by @thisjeremiah in #4029
- Rename "createSliceWithThunks" and "createThunkSlice" to "createAppSlice" by @EskiMojo14 in #4035
- Bump Vitest to latest version by @aryaemami59 in #4047
- fix inference and order of async thunk generics by @EskiMojo14 in #4061
- Fix markdown links on Next.js page by @DmitryScaletta in #4069
- Introduce pre-typed listener middleware via
listenerMiddleware.withTypes<RootState, AppDispatch>()method by @aryaemami59 in #4049 - Add
.withTypestocreateDraftSafeSelectorby @aryaemami59 in #4080 - Task/remove all settled by @bever1337 in #3917
- Allow customising createSelector instance used by RTKQ by @EskiMojo14 in #4048
- cacheLifecycle fix for fixedCacheKey in mutationThunk by @riqts in #4082
- avoid relying on
thisin createSlice by @EskiMojo14 in #4071 - documentation: fix grammar in redux and redux toolkit comparison by @untilhamza in #4086
- Update
tsconfig.typetests.jsonto include all TS files by @aryaemami59 in #4091 - Remove abort event listner for AbortController by @Fonger in #3951
- Docs: unused value 'isRejected' in code snippet by @alphonsotran in #3301
- fix/kitchen-sink-isAuthenticated: fixed isAuthenticated state change on login fullfilled by @shrijan00003 in #3588
- Fix
composeWithDevToolsspy by @aryaemami59 in #4093 - require queryFn meta to match base query by @EskiMojo14 in #4098
- Added 'SafePromise' branded Promises for createAsyncThunk by @JoshuaKGoldberg in #4102
- Add React Native demo app to CI workflow by @aryaemami59 in #3984
- Modernize unit test setup by @aryaemami59 in #4114
- Add missing reducer field to ConfigureStoreOptions docs by @nickgirardo in #4116
- Option for queries to pause polling when unfocused by @riqts in #4055
New Contributors
- @suspiciousRaccoon made their first contribution in #3611
- @fatihgnc made their first contribution in #4013
- @thisjeremiah made their first contribution in #4029
- @riqts made their first contribution in #4082
- @untilhamza made their first contribution in #4086
- @Fonger made their first contribution in #3951
- @alphonsotran made their first contribution in #3301
- @shrijan00003 made their first contribution in #3588
- @JoshuaKGoldberg made their first contribution in #4102
- @nickgirardo made their first contribution in #4116
Full Changelog: v2.0.1...v2.1.0
v2.0.0
This major release :
- Removes the deprecated object syntax from
createSliceandcreateReducer - Removes other deprecated options
- Updates the
middlewareandenhancersoptions ofconfigureStoreto 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
combineSlicesmethod 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
autoBatchEnhancertoconfigureStoreby 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/toolkitChangelog
Object syntax for createSlice.extraReducers and createReducer removed
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/**/*.tsWe 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.
Standalone getDefaultMiddleware and getType removed
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.mj...