Skip to content

Commit

Permalink
Merge pull request reduxjs#2835 from reduxjs/feature/rework-sqa
Browse files Browse the repository at this point in the history
  • Loading branch information
markerikson authored Oct 28, 2022
2 parents e0a8bd5 + d8d1af6 commit 17d1789
Show file tree
Hide file tree
Showing 4 changed files with 105 additions and 12 deletions.
16 changes: 15 additions & 1 deletion packages/toolkit/src/query/createApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -249,8 +249,22 @@ export function buildCreateApi<Modules extends [Module<any>, ...Module<any>[]]>(
serializeQueryArgs(queryArgsApi) {
let finalSerializeQueryArgs = defaultSerializeQueryArgs
if ('serializeQueryArgs' in queryArgsApi.endpointDefinition) {
finalSerializeQueryArgs =
const endpointSQA =
queryArgsApi.endpointDefinition.serializeQueryArgs!
finalSerializeQueryArgs = (queryArgsApi) => {
const initialResult = endpointSQA(queryArgsApi)
if (typeof initialResult === 'string') {
// If the user function returned a string, use it as-is
return initialResult
} else {
// Assume they returned an object (such as a subset of the original
// query args) or a primitive, and serialize it ourselves
return defaultSerializeQueryArgs({
...queryArgsApi,
queryArgs: initialResult,
})
}
}
} else if (options.serializeQueryArgs) {
finalSerializeQueryArgs = options.serializeQueryArgs
}
Expand Down
4 changes: 2 additions & 2 deletions packages/toolkit/src/query/defaultSerializeQueryArgs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,11 @@ export const defaultSerializeQueryArgs: SerializeQueryArgs<any> = ({
)})`
}

export type SerializeQueryArgs<QueryArgs> = (_: {
export type SerializeQueryArgs<QueryArgs, ReturnType = string> = (_: {
queryArgs: QueryArgs
endpointDefinition: EndpointDefinition<any, any, any, any>
endpointName: string
}) => string
}) => ReturnType

export type InternalSerializeQueryArgs = (_: {
queryArgs: any
Expand Down
29 changes: 20 additions & 9 deletions packages/toolkit/src/query/endpointDefinitions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -330,10 +330,13 @@ export interface QueryExtraOptions<
invalidatesTags?: never

/**
* Can be provided to return a custom cache key string based on the provided arguments.
* Can be provided to return a custom cache key value based on the query arguments.
*
* This is primarily intended for cases where a non-serializable value is passed as part of the query arg object and should be excluded from the cache key. It may also be used for cases where an endpoint should only have a single cache entry, such as an infinite loading / pagination implementation.
*
* Unlike the `createApi` version which can _only_ return a string, this per-endpoint option can also return an an object, number, or boolean. If it returns a string, that value will be used as the cache key directly. If it returns an object / number / boolean, that value will be passed to the built-in `defaultSerializeQueryArgs`. This simplifies the use case of stripping out args you don't want included in the cache key.
*
*
* @example
*
* ```ts
Expand Down Expand Up @@ -362,13 +365,18 @@ export interface QueryExtraOptions<
* // highlight-start
* serializeQueryArgs: ({ queryArgs, endpointDefinition, endpointName }) => {
* const { id } = queryArgs
* // You can use `defaultSerializeQueryArgs` to do the work:
* return defaultSerializeQueryArgs({
* endpointName,
* queryArgs: { id },
* endpointDefinition
* })
* // Or alternately, create a string yourself:
* // This can return a string, an object, a number, or a boolean.
* // If it returns an object, number or boolean, that value
* // will be serialized automatically via `defaultSerializeQueryArgs`
* return { id } // omit `client` from the cache key
*
* // Alternately, you can use `defaultSerializeQueryArgs` yourself:
* // return defaultSerializeQueryArgs({
* // endpointName,
* // queryArgs: { id },
* // endpointDefinition
* // })
* // Or create and return a string yourself:
* // return `getPost(${id})`
* },
* // highlight-end
Expand All @@ -377,7 +385,10 @@ export interface QueryExtraOptions<
*})
* ```
*/
serializeQueryArgs?: SerializeQueryArgs<QueryArg>
serializeQueryArgs?: SerializeQueryArgs<
QueryArg,
string | number | boolean | Record<any, any>
>

/**
* Can be provided to merge an incoming response value into the current cache data.
Expand Down
68 changes: 68 additions & 0 deletions packages/toolkit/src/query/tests/createApi.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -854,6 +854,16 @@ describe('custom serializeQueryArgs per endpoint', () => {

const serializer1 = jest.fn(customArgsSerializer)

interface MyApiClient {
fetchPost: (id: string) => Promise<SuccessResponse>
}

const dummyClient: MyApiClient = {
async fetchPost(id) {
return { value: 'success' }
},
}

const api = createApi({
baseQuery: fetchBaseQuery({ baseUrl: 'https://example.com' }),
serializeQueryArgs: ({ endpointName, queryArgs }) =>
Expand All @@ -866,6 +876,34 @@ describe('custom serializeQueryArgs per endpoint', () => {
query: (arg) => `${arg}`,
serializeQueryArgs: serializer1,
}),
queryWithCustomObjectSerializer: build.query<
SuccessResponse,
{ id: number; client: MyApiClient }
>({
query: (arg) => `${arg.id}`,
serializeQueryArgs: ({
endpointDefinition,
endpointName,
queryArgs,
}) => {
const { id } = queryArgs
return { id }
},
}),
queryWithCustomNumberSerializer: build.query<
SuccessResponse,
{ id: number; client: MyApiClient }
>({
query: (arg) => `${arg.id}`,
serializeQueryArgs: ({
endpointDefinition,
endpointName,
queryArgs,
}) => {
const { id } = queryArgs
return id
},
}),
listItems: build.query<string[], number>({
query: (pageNumber) => `/listItems?page=${pageNumber}`,
serializeQueryArgs: ({ endpointName }) => {
Expand Down Expand Up @@ -931,6 +969,36 @@ describe('custom serializeQueryArgs per endpoint', () => {
).toBeTruthy()
})

test('Serializes a returned object for query args', async () => {
await storeRef.store.dispatch(
api.endpoints.queryWithCustomObjectSerializer.initiate({
id: 42,
client: dummyClient,
})
)

expect(
storeRef.store.getState().api.queries[
'queryWithCustomObjectSerializer({"id":42})'
]
).toBeTruthy()
})

test('Serializes a returned primitive for query args', async () => {
await storeRef.store.dispatch(
api.endpoints.queryWithCustomNumberSerializer.initiate({
id: 42,
client: dummyClient,
})
)

expect(
storeRef.store.getState().api.queries[
'queryWithCustomNumberSerializer(42)'
]
).toBeTruthy()
})

test('serializeQueryArgs + merge allows refetching as args change with same cache key', async () => {
const allItems = ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'i']
const PAGE_SIZE = 3
Expand Down

0 comments on commit 17d1789

Please sign in to comment.