Skip to content
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

Rewrite providedTags handling for better perf #4910

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
56 changes: 24 additions & 32 deletions packages/toolkit/src/query/core/apiState.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import type { BaseQueryError } from '../baseQueryTypes'
import type {
BaseEndpointDefinition,
EndpointDefinitions,
FullTagDescription,
InfiniteQueryDefinition,
MutationDefinition,
PageParamFrom,
Expand All @@ -15,14 +16,8 @@ import type { Id, WithRequiredProp } from '../tsHelpers'
export type QueryCacheKey = string & { _type: 'queryCacheKey' }
export type QuerySubstateIdentifier = { queryCacheKey: QueryCacheKey }
export type MutationSubstateIdentifier =
| {
requestId: string
fixedCacheKey?: string
}
| {
requestId?: string
fixedCacheKey: string
}
| { requestId: string; fixedCacheKey?: string }
| { requestId?: string; fixedCacheKey: string }

export type RefetchConfigOptions = {
refetchOnMountOrArgChange: boolean | number
Expand Down Expand Up @@ -225,18 +220,15 @@ export type QuerySubState<
D extends BaseEndpointDefinition<any, any, any>,
DataType = ResultTypeFrom<D>,
> = Id<
| ({
status: QueryStatus.fulfilled
} & WithRequiredProp<
| ({ status: QueryStatus.fulfilled } & WithRequiredProp<
BaseQuerySubState<D, DataType>,
'data' | 'fulfilledTimeStamp'
> & { error: undefined })
| ({
status: QueryStatus.pending
} & BaseQuerySubState<D, DataType>)
| ({
status: QueryStatus.rejected
} & WithRequiredProp<BaseQuerySubState<D, DataType>, 'error'>)
| ({ status: QueryStatus.pending } & BaseQuerySubState<D, DataType>)
| ({ status: QueryStatus.rejected } & WithRequiredProp<
BaseQuerySubState<D, DataType>,
'error'
>)
| {
status: QueryStatus.uninitialized
originalArgs?: undefined
Expand Down Expand Up @@ -274,18 +266,17 @@ type BaseMutationSubState<D extends BaseEndpointDefinition<any, any, any>> = {
}

export type MutationSubState<D extends BaseEndpointDefinition<any, any, any>> =
| (({
status: QueryStatus.fulfilled
} & WithRequiredProp<
| (({ status: QueryStatus.fulfilled } & WithRequiredProp<
BaseMutationSubState<D>,
'data' | 'fulfilledTimeStamp'
>) & { error: undefined })
| (({
status: QueryStatus.pending
} & BaseMutationSubState<D>) & { data?: undefined })
| ({
status: QueryStatus.rejected
} & WithRequiredProp<BaseMutationSubState<D>, 'error'>)
| (({ status: QueryStatus.pending } & BaseMutationSubState<D>) & {
data?: undefined
})
| ({ status: QueryStatus.rejected } & WithRequiredProp<
BaseMutationSubState<D>,
'error'
>)
| {
requestId?: undefined
status: QueryStatus.uninitialized
Expand All @@ -309,10 +300,13 @@ export type CombinedState<
}

export type InvalidationState<TagTypes extends string> = {
[_ in TagTypes]: {
[id: string]: Array<QueryCacheKey>
[id: number]: Array<QueryCacheKey>
tags: {
[_ in TagTypes]: {
[id: string]: Array<QueryCacheKey>
[id: number]: Array<QueryCacheKey>
}
}
keys: Record<QueryCacheKey, Array<FullTagDescription<any>>>
}

export type QueryState<D extends EndpointDefinitions> = {
Expand Down Expand Up @@ -346,6 +340,4 @@ export type RootState<
Definitions extends EndpointDefinitions,
TagTypes extends string,
ReducerPath extends string,
> = {
[P in ReducerPath]: CombinedState<Definitions, TagTypes, P>
}
> = { [P in ReducerPath]: CombinedState<Definitions, TagTypes, P> }
7 changes: 2 additions & 5 deletions packages/toolkit/src/query/core/buildSelectors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -199,10 +199,7 @@ export function buildSelectors<
function withRequestFlags<T extends { status: QueryStatus }>(
substate: T,
): T & RequestStatusFlags {
return {
...substate,
...getRequestStatusFlags(substate.status),
}
return { ...substate, ...getRequestStatusFlags(substate.status) }
}

function selectApiState(rootState: RootState) {
Expand Down Expand Up @@ -344,7 +341,7 @@ export function buildSelectors<
const apiState = state[reducerPath]
const toInvalidate = new Set<QueryCacheKey>()
for (const tag of tags.filter(isNotNullish).map(expandTagDescription)) {
const provided = apiState.provided[tag.type]
const provided = apiState.provided.tags[tag.type]
if (!provided) {
continue
}
Expand Down
157 changes: 86 additions & 71 deletions packages/toolkit/src/query/core/buildSlice.ts
Original file line number Diff line number Diff line change
Expand Up @@ -217,10 +217,7 @@ export function buildSlice({

function writeFulfilledCacheEntry(
draft: QueryState<any>,
meta: {
arg: QueryThunkArg
requestId: string
} & {
meta: { arg: QueryThunkArg; requestId: string } & {
fulfilledTimeStamp: number
baseQueryMeta: unknown
},
Expand Down Expand Up @@ -297,11 +294,7 @@ export function buildSlice({
action: PayloadAction<
ProcessedQueryUpsertEntry[],
string,
{
RTK_autoBatch: boolean
requestId: string
timestamp: number
}
{ RTK_autoBatch: boolean; requestId: string; timestamp: number }
>,
) {
for (const entry of action.payload) {
Expand Down Expand Up @@ -488,67 +481,66 @@ export function buildSlice({
| ReturnType<ReturnType<InfiniteQueryThunk<any>>>
>

const initialInvalidationState: InvalidationState<string> = {
tags: {},
keys: {},
}

const invalidationSlice = createSlice({
name: `${reducerPath}/invalidation`,
initialState: initialState as InvalidationState<string>,
initialState: initialInvalidationState,
reducers: {
updateProvidedBy: {
reducer(
draft,
action: PayloadAction<{
queryCacheKey: QueryCacheKey
providedTags: readonly FullTagDescription<string>[]
}>,
action: PayloadAction<
Array<{
queryCacheKey: QueryCacheKey
providedTags: readonly FullTagDescription<string>[]
}>
>,
) {
const { queryCacheKey, providedTags } = action.payload
for (const { queryCacheKey, providedTags } of action.payload) {
removeCacheKeyFromTags(draft, queryCacheKey)

for (const tagTypeSubscriptions of Object.values(draft)) {
for (const idSubscriptions of Object.values(tagTypeSubscriptions)) {
const foundAt = idSubscriptions.indexOf(queryCacheKey)
if (foundAt !== -1) {
idSubscriptions.splice(foundAt, 1)
for (const { type, id } of providedTags) {
const subscribedQueries = ((draft.tags[type] ??= {})[
id || '__internal_without_id'
] ??= [])
const alreadySubscribed =
subscribedQueries.includes(queryCacheKey)
if (!alreadySubscribed) {
subscribedQueries.push(queryCacheKey)
}
}
}

for (const { type, id } of providedTags) {
const subscribedQueries = ((draft[type] ??= {})[
id || '__internal_without_id'
] ??= [])
const alreadySubscribed = subscribedQueries.includes(queryCacheKey)
if (!alreadySubscribed) {
subscribedQueries.push(queryCacheKey)
}
// Remove readonly from the providedTags array
draft.keys[queryCacheKey] =
providedTags as FullTagDescription<string>[]
}
},
prepare: prepareAutoBatched<{
queryCacheKey: QueryCacheKey
providedTags: readonly FullTagDescription<string>[]
}>(),
prepare:
prepareAutoBatched<
Array<{
queryCacheKey: QueryCacheKey
providedTags: readonly FullTagDescription<string>[]
}>
>(),
},
},
extraReducers(builder) {
builder
.addCase(
querySlice.actions.removeQueryResult,
(draft, { payload: { queryCacheKey } }) => {
for (const tagTypeSubscriptions of Object.values(draft)) {
for (const idSubscriptions of Object.values(
tagTypeSubscriptions,
)) {
const foundAt = idSubscriptions.indexOf(queryCacheKey)
if (foundAt !== -1) {
idSubscriptions.splice(foundAt, 1)
}
}
}
removeCacheKeyFromTags(draft, queryCacheKey)
},
)
.addMatcher(hasRehydrationInfo, (draft, action) => {
const { provided } = extractRehydrationInfo(action)!
for (const [type, incomingTags] of Object.entries(provided)) {
for (const [id, cacheKeys] of Object.entries(incomingTags)) {
const subscribedQueries = ((draft[type] ??= {})[
const subscribedQueries = ((draft.tags[type] ??= {})[
id || '__internal_without_id'
] ??= [])
for (const queryCacheKey of cacheKeys) {
Expand All @@ -564,48 +556,71 @@ export function buildSlice({
.addMatcher(
isAnyOf(isFulfilled(queryThunk), isRejectedWithValue(queryThunk)),
(draft, action) => {
writeProvidedTagsForQuery(draft, action)
writeProvidedTagsForQueries(draft, [action])
},
)
.addMatcher(
querySlice.actions.cacheEntriesUpserted.match,
(draft, action) => {
for (const { queryDescription: arg, value } of action.payload) {
const action: CalculateProvidedByAction = {
type: 'UNKNOWN',
payload: value,
meta: {
requestStatus: 'fulfilled',
requestId: 'UNKNOWN',
arg,
},
}

writeProvidedTagsForQuery(draft, action)
}
const mockActions: CalculateProvidedByAction[] = action.payload.map(
({ queryDescription, value }) => {
return {
type: 'UNKNOWN',
payload: value,
meta: {
requestStatus: 'fulfilled',
requestId: 'UNKNOWN',
arg: queryDescription,
},
}
},
)
writeProvidedTagsForQueries(draft, mockActions)
},
)
},
})

function writeProvidedTagsForQuery(
function removeCacheKeyFromTags(
draft: InvalidationState<any>,
queryCacheKey: QueryCacheKey,
) {
const existingTags = draft.keys[queryCacheKey] ?? []

// Delete this cache key from any existing tags that may have provided it
for (const tag of existingTags) {
const tagType = tag.type
const tagId = tag.id ?? '__internal_without_id'
const tagSubscriptions = draft.tags[tagType]?.[tagId]

if (tagSubscriptions) {
draft.tags[tagType][tagId] = tagSubscriptions.filter(
(qc) => qc !== queryCacheKey,
)
}
}

delete draft.keys[queryCacheKey]
}

function writeProvidedTagsForQueries(
draft: InvalidationState<string>,
action: CalculateProvidedByAction,
actions: CalculateProvidedByAction[],
) {
const providedTags = calculateProvidedByThunk(
action,
'providesTags',
definitions,
assertTagType,
)
const { queryCacheKey } = action.meta.arg
const providedByEntries = actions.map((action) => {
const providedTags = calculateProvidedByThunk(
action,
'providesTags',
definitions,
assertTagType,
)
const { queryCacheKey } = action.meta.arg
return { queryCacheKey, providedTags }
})

invalidationSlice.caseReducers.updateProvidedBy(
draft,
invalidationSlice.actions.updateProvidedBy({
queryCacheKey,
providedTags,
}),
invalidationSlice.actions.updateProvidedBy(providedByEntries),
)
}

Expand Down
Loading