Skip to content

Commit

Permalink
Ensure createConnectQueryKey enforces type on input
Browse files Browse the repository at this point in the history
Signed-off-by: Paul Sachs <[email protected]>
  • Loading branch information
paul-sachs committed Dec 9, 2024
1 parent e462d8c commit a4d3773
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 45 deletions.
3 changes: 1 addition & 2 deletions packages/connect-query-core/src/connect-query-key.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -123,12 +123,11 @@ describe("createConnectQueryKey", () => {
expect(key[1].methodName).toBeUndefined();
});

// eslint-disable-next-line vitest/expect-expect -- using ts-expect-error
it("cannot except invalid input", () => {
// @ts-expect-error(2322) cannot create a key with invalid input
createConnectQueryKey({
schema: ElizaService.method.say,
input: {
// @ts-expect-error(2322) cannot create a key with invalid input
sentence: 1,
},
cardinality: undefined,
Expand Down
91 changes: 48 additions & 43 deletions packages/connect-query-core/src/connect-query-key.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,9 @@
// limitations under the License.

import type {
DescMessage,
DescMethod,
DescMethodUnary,
DescService,
MessageInitShape,
} from "@bufbuild/protobuf";
Expand Down Expand Up @@ -72,46 +74,46 @@ export type ConnectQueryKey = [
},
];

type KeyParams<Desc extends DescMethod | DescService> = Desc extends DescMethod
? {
/**
* Set `serviceName` and `methodName` in the key.
*/
schema: Desc;
/**
* Set `input` in the key:
* - If a SkipToken is provided, `input` is "skipped".
* - If an init shape is provided, `input` is set to a message key.
* - If omitted or undefined, `input` is not set in the key.
*/
input?: MessageInitShape<Desc["input"]> | SkipToken | undefined;
/**
* Set `transport` in the key.
*/
transport?: Transport;
/**
* Set `cardinality` in the key - undefined is used for filters to match both finite and infinite queries.
*/
cardinality: "finite" | "infinite" | undefined;
/**
* If omit the field with this name from the key for infinite queries.
*/
pageParamKey?: keyof MessageInitShape<Desc["input"]>;
}
: {
/**
* Set `serviceName` in the key, and omit `methodName`.
*/
schema: Desc;
/**
* Set `transport` in the key.
*/
transport?: Transport;
/**
* Set `cardinality` in the key - undefined is used for filters to match both finite and infinite queries.
*/
cardinality: "finite" | "infinite" | undefined;
};
type KeyParamsForMethod<Desc extends DescMethod> = {
/**
* Set `serviceName` and `methodName` in the key.
*/
schema: Desc;
/**
* Set `input` in the key:
* - If a SkipToken is provided, `input` is "skipped".
* - If an init shape is provided, `input` is set to a message key.
* - If omitted or undefined, `input` is not set in the key.
*/
input?: MessageInitShape<Desc["input"]> | SkipToken | undefined;
/**
* Set `transport` in the key.
*/
transport?: Transport;
/**
* Set `cardinality` in the key - undefined is used for filters to match both finite and infinite queries.
*/
cardinality: "finite" | "infinite" | undefined;
/**
* If omit the field with this name from the key for infinite queries.
*/
pageParamKey?: keyof MessageInitShape<Desc["input"]>;
};

type KeyParamsForService<Desc extends DescService> = {
/**
* Set `serviceName` in the key, and omit `methodName`.
*/
schema: Desc;
/**
* Set `transport` in the key.
*/
transport?: Transport;
/**
* Set `cardinality` in the key - undefined is used for filters to match both finite and infinite queries.
*/
cardinality: "finite" | "infinite" | undefined;
};

/**
* TanStack Query manages query caching for you based on query keys. In Connect Query, keys are structured, and can easily be created using this factory function.
Expand Down Expand Up @@ -151,9 +153,12 @@ type KeyParams<Desc extends DescMethod | DescService> = Desc extends DescMethod
* @see ConnectQueryKey for information on the components of Connect-Query's keys.
*/
export function createConnectQueryKey<
Desc extends DescMethod | DescService,
Params extends KeyParams<Desc>,
>(params: Params): ConnectQueryKey {
I extends DescMessage,
O extends DescMessage,
Desc extends DescService,
>(
params: KeyParamsForMethod<DescMethodUnary<I, O>> | KeyParamsForService<Desc>,
): ConnectQueryKey {
const props: ConnectQueryKey[1] =
params.schema.kind == "rpc"
? {
Expand Down
13 changes: 13 additions & 0 deletions packages/connect-query-core/src/create-query-options.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -58,4 +58,17 @@ describe("createQueryOptions", () => {
);
expect(opt.queryKey).toStrictEqual(want);
});

it("ensures type safety of parameters", () => {
// @ts-expect-error(2322) cannot provide invalid parameters
createQueryOptions(
sayMethodDescriptor,
{
sentence: 1,
},
{
transport: mockedElizaTransport,
},
);
});
});

0 comments on commit a4d3773

Please sign in to comment.