@orpc/react 0.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/dist/index.js +635 -0
  2. package/dist/src/general-hooks.d.ts +24 -0
  3. package/dist/src/general-hooks.d.ts.map +1 -0
  4. package/dist/src/general-utils.d.ts +41 -0
  5. package/dist/src/general-utils.d.ts.map +1 -0
  6. package/dist/src/index.d.ts +9 -0
  7. package/dist/src/index.d.ts.map +1 -0
  8. package/dist/src/orpc-path.d.ts +5 -0
  9. package/dist/src/orpc-path.d.ts.map +1 -0
  10. package/dist/src/procedure-hooks.d.ts +31 -0
  11. package/dist/src/procedure-hooks.d.ts.map +1 -0
  12. package/dist/src/procedure-utils.d.ts +35 -0
  13. package/dist/src/procedure-utils.d.ts.map +1 -0
  14. package/dist/src/react-context.d.ts +13 -0
  15. package/dist/src/react-context.d.ts.map +1 -0
  16. package/dist/src/react-hooks.d.ts +22 -0
  17. package/dist/src/react-hooks.d.ts.map +1 -0
  18. package/dist/src/react-utils.d.ts +22 -0
  19. package/dist/src/react-utils.d.ts.map +1 -0
  20. package/dist/src/react.d.ts +21 -0
  21. package/dist/src/react.d.ts.map +1 -0
  22. package/dist/src/tanstack-key.d.ts +15 -0
  23. package/dist/src/tanstack-key.d.ts.map +1 -0
  24. package/dist/src/tanstack-query.d.ts +19 -0
  25. package/dist/src/tanstack-query.d.ts.map +1 -0
  26. package/dist/src/types.d.ts +5 -0
  27. package/dist/src/types.d.ts.map +1 -0
  28. package/dist/src/use-queries/builder.d.ts +20 -0
  29. package/dist/src/use-queries/builder.d.ts.map +1 -0
  30. package/dist/src/use-queries/builders.d.ts +19 -0
  31. package/dist/src/use-queries/builders.d.ts.map +1 -0
  32. package/dist/src/use-queries/hook.d.ts +16 -0
  33. package/dist/src/use-queries/hook.d.ts.map +1 -0
  34. package/dist/tsconfig.tsbuildinfo +1 -0
  35. package/package.json +56 -0
  36. package/src/general-hooks.test-d.ts +151 -0
  37. package/src/general-hooks.test.tsx +232 -0
  38. package/src/general-hooks.ts +100 -0
  39. package/src/general-utils.test-d.ts +454 -0
  40. package/src/general-utils.test.tsx +818 -0
  41. package/src/general-utils.ts +397 -0
  42. package/src/index.ts +8 -0
  43. package/src/orpc-path.test-d.ts +13 -0
  44. package/src/orpc-path.test.ts +12 -0
  45. package/src/orpc-path.ts +24 -0
  46. package/src/procedure-hooks.test-d.ts +321 -0
  47. package/src/procedure-hooks.test.tsx +388 -0
  48. package/src/procedure-hooks.ts +271 -0
  49. package/src/procedure-utils.test-d.ts +476 -0
  50. package/src/procedure-utils.test.tsx +330 -0
  51. package/src/procedure-utils.ts +315 -0
  52. package/src/react-context.ts +43 -0
  53. package/src/react-hooks.ts +102 -0
  54. package/src/react-utils.ts +110 -0
  55. package/src/react.test-d.ts +89 -0
  56. package/src/react.test.tsx +102 -0
  57. package/src/react.tsx +80 -0
  58. package/src/tanstack-key.test-d.ts +35 -0
  59. package/src/tanstack-key.test.ts +62 -0
  60. package/src/tanstack-key.ts +64 -0
  61. package/src/tanstack-query.ts +27 -0
  62. package/src/types.ts +7 -0
  63. package/src/use-queries/builder.test-d.ts +29 -0
  64. package/src/use-queries/builder.test.ts +25 -0
  65. package/src/use-queries/builder.ts +72 -0
  66. package/src/use-queries/builders.test-d.ts +30 -0
  67. package/src/use-queries/builders.test.tsx +29 -0
  68. package/src/use-queries/builders.ts +101 -0
  69. package/src/use-queries/hook.test-d.ts +64 -0
  70. package/src/use-queries/hook.test.tsx +89 -0
  71. package/src/use-queries/hook.ts +57 -0
@@ -0,0 +1,29 @@
1
+ import type { Promisable } from '@orpc/server'
2
+ import { orpcClient } from '../../tests/orpc'
3
+ import { createUseQueriesBuilder } from './builder'
4
+
5
+ it('createUseQueriesBuilder', () => {
6
+ const builder = createUseQueriesBuilder({
7
+ client: orpcClient.user.find,
8
+ path: ['user', 'find'],
9
+ })
10
+
11
+ expectTypeOf<Parameters<typeof builder>[0]>().toEqualTypeOf<{
12
+ id: string
13
+ }>()
14
+
15
+ const result = builder({ id: '123' }, { throwOnError: true })
16
+
17
+ builder({ id: '123' })
18
+ // @ts-expect-error invalid input
19
+ builder({ id: 123 })
20
+
21
+ if (typeof result.queryFn === 'function') {
22
+ expectTypeOf(result.queryFn({} as any)).toEqualTypeOf<
23
+ Promisable<{
24
+ id: string
25
+ name: string
26
+ }>
27
+ >()
28
+ }
29
+ })
@@ -0,0 +1,25 @@
1
+ import { orpcClient } from '../../tests/orpc'
2
+ import { createUseQueriesBuilder } from './builder'
3
+
4
+ it('createUseQueriesBuilder', async () => {
5
+ const builder = createUseQueriesBuilder({
6
+ client: orpcClient.user.find,
7
+ path: ['user', 'find'],
8
+ })
9
+
10
+ const options = builder({ id: '123' })
11
+
12
+ expect(options.queryKey).toEqual([
13
+ ['user', 'find'],
14
+ { input: { id: '123' }, type: 'query' },
15
+ ])
16
+
17
+ expect(options.queryFn).toBeInstanceOf(Function)
18
+
19
+ const result = await (options as any).queryFn({} as any)
20
+
21
+ expect(result).toEqual({
22
+ id: '123',
23
+ name: 'name-123',
24
+ })
25
+ })
@@ -0,0 +1,72 @@
1
+ import type { ProcedureClient } from '@orpc/client'
2
+ import type { Schema, SchemaInput, SchemaOutput } from '@orpc/contract'
3
+ import type {} from '@orpc/server'
4
+ import type { SetOptional } from '@orpc/shared'
5
+ import type {
6
+ DefaultError,
7
+ OmitKeyof,
8
+ QueriesPlaceholderDataFunction,
9
+ QueryKey,
10
+ UseQueryOptions,
11
+ } from '@tanstack/react-query'
12
+ import { getQueryKeyFromPath } from '../tanstack-key'
13
+
14
+ type UseQueryOptionsForUseQueries<
15
+ TQueryFnData,
16
+ TError = DefaultError,
17
+ TData = TQueryFnData,
18
+ TQueryKey extends QueryKey = QueryKey,
19
+ > = OmitKeyof<
20
+ UseQueryOptions<TQueryFnData, TError, TData, TQueryKey>,
21
+ 'placeholderData'
22
+ > & {
23
+ placeholderData?: TQueryFnData | QueriesPlaceholderDataFunction<TQueryFnData>
24
+ }
25
+
26
+ export interface UseQueriesBuilder<
27
+ TInputSchema extends Schema,
28
+ TOutputSchema extends Schema,
29
+ THandlerOutput extends SchemaOutput<TOutputSchema>,
30
+ > {
31
+ (
32
+ input: SchemaInput<TInputSchema>,
33
+ options?: SetOptional<
34
+ UseQueryOptionsForUseQueries<SchemaOutput<TOutputSchema, THandlerOutput>>,
35
+ 'queryFn' | 'queryKey'
36
+ >,
37
+ ): UseQueryOptionsForUseQueries<SchemaOutput<TOutputSchema, THandlerOutput>>
38
+ }
39
+
40
+ export interface CreateUseQueriesBuilderOptions<
41
+ TInputSchema extends Schema,
42
+ TOutputSchema extends Schema,
43
+ THandlerOutput extends SchemaOutput<TOutputSchema>,
44
+ > {
45
+ client: ProcedureClient<TInputSchema, TOutputSchema, THandlerOutput>
46
+
47
+ /**
48
+ * The path of procedure on server
49
+ */
50
+ path: string[]
51
+ }
52
+
53
+ export function createUseQueriesBuilder<
54
+ TInputSchema extends Schema = undefined,
55
+ TOutputSchema extends Schema = undefined,
56
+ THandlerOutput extends
57
+ SchemaOutput<TOutputSchema> = SchemaOutput<TOutputSchema>,
58
+ >(
59
+ options: CreateUseQueriesBuilderOptions<
60
+ TInputSchema,
61
+ TOutputSchema,
62
+ THandlerOutput
63
+ >,
64
+ ): UseQueriesBuilder<TInputSchema, TOutputSchema, THandlerOutput> {
65
+ return (input, options_) => {
66
+ return {
67
+ queryKey: getQueryKeyFromPath(options.path, { input, type: 'query' }),
68
+ queryFn: () => options.client(input),
69
+ ...options_,
70
+ }
71
+ }
72
+ }
@@ -0,0 +1,30 @@
1
+ import { orpcClient } from '../../tests/orpc'
2
+ import { createUseQueriesBuilder } from './builder'
3
+ import { createUseQueriesBuilders } from './builders'
4
+
5
+ it('createUseQueriesBuilders', () => {
6
+ const builder = createUseQueriesBuilders({
7
+ client: orpcClient,
8
+ })
9
+
10
+ expectTypeOf(builder.user.find).toEqualTypeOf(
11
+ createUseQueriesBuilder({
12
+ client: orpcClient.user.find,
13
+ path: ['user', 'find'],
14
+ }),
15
+ )
16
+
17
+ expectTypeOf(builder.user.list).toEqualTypeOf(
18
+ createUseQueriesBuilder({
19
+ client: orpcClient.user.list,
20
+ path: ['user', 'list'],
21
+ }),
22
+ )
23
+
24
+ expectTypeOf(builder.ping).toEqualTypeOf(
25
+ createUseQueriesBuilder({
26
+ client: orpcClient.ping,
27
+ path: ['ping'],
28
+ }),
29
+ )
30
+ })
@@ -0,0 +1,29 @@
1
+ import { orpcClient } from '../../tests/orpc'
2
+ import { createUseQueriesBuilder } from './builder'
3
+ import { createUseQueriesBuilders } from './builders'
4
+
5
+ it('createUseQueriesBuilders', async () => {
6
+ const builder = createUseQueriesBuilders({
7
+ client: orpcClient,
8
+ })
9
+
10
+ const e1 = builder.user.find({ id: '123' })
11
+ const a1 = createUseQueriesBuilder({
12
+ client: orpcClient.user.find,
13
+ path: ['user', 'find'],
14
+ })({ id: '123' })
15
+
16
+ expect(e1.queryKey).toEqual(a1.queryKey)
17
+
18
+ expect(await (e1 as any).queryFn()).toEqual(await (a1 as any).queryFn())
19
+
20
+ const e2 = builder.user.list({})
21
+ const a2 = createUseQueriesBuilder({
22
+ client: orpcClient.user.list,
23
+ path: ['user', 'list'],
24
+ })({})
25
+
26
+ expect(e2.queryKey).toEqual(a2.queryKey)
27
+
28
+ expect(await (e2 as any).queryFn()).toEqual(await (a2 as any).queryFn())
29
+ })
@@ -0,0 +1,101 @@
1
+ import type {
2
+ RouterClientWithContractRouter,
3
+ RouterClientWithRouter,
4
+ } from '@orpc/client'
5
+ import type {
6
+ ContractProcedure,
7
+ ContractRouter,
8
+ SchemaOutput,
9
+ } from '@orpc/contract'
10
+ import type { Procedure, Router } from '@orpc/server'
11
+ import type {} from '@tanstack/react-query'
12
+ import { type UseQueriesBuilder, createUseQueriesBuilder } from './builder'
13
+
14
+ export type UseQueriesBuildersWithContractRouter<
15
+ TRouter extends ContractRouter,
16
+ > = {
17
+ [K in keyof TRouter]: TRouter[K] extends ContractProcedure<
18
+ infer UInputSchema,
19
+ infer UOutputSchema
20
+ >
21
+ ? UseQueriesBuilder<
22
+ UInputSchema,
23
+ UOutputSchema,
24
+ SchemaOutput<UOutputSchema>
25
+ >
26
+ : TRouter[K] extends ContractRouter
27
+ ? UseQueriesBuildersWithContractRouter<TRouter[K]>
28
+ : never
29
+ }
30
+
31
+ export type UseQueriesBuildersWithRouter<TRouter extends Router<any>> = {
32
+ [K in keyof TRouter]: TRouter[K] extends Procedure<
33
+ any,
34
+ any,
35
+ infer UInputSchema,
36
+ infer UOutputSchema,
37
+ infer UHandlerOutput
38
+ >
39
+ ? UseQueriesBuilder<UInputSchema, UOutputSchema, UHandlerOutput>
40
+ : TRouter[K] extends Router<any>
41
+ ? UseQueriesBuildersWithRouter<TRouter[K]>
42
+ : never
43
+ }
44
+
45
+ export interface CreateUseQueriesBuildersOptions<
46
+ TRouter extends Router<any> | ContractRouter,
47
+ > {
48
+ client: TRouter extends Router<any>
49
+ ? RouterClientWithRouter<TRouter>
50
+ : TRouter extends ContractRouter
51
+ ? RouterClientWithContractRouter<TRouter>
52
+ : never
53
+
54
+ /**
55
+ * The path of router on server
56
+ */
57
+ path?: string[]
58
+ }
59
+
60
+ export function createUseQueriesBuilders<
61
+ TRouter extends Router<any> | ContractRouter,
62
+ >(
63
+ options: CreateUseQueriesBuildersOptions<TRouter>,
64
+ ): TRouter extends Router<any>
65
+ ? UseQueriesBuildersWithRouter<TRouter>
66
+ : TRouter extends ContractRouter
67
+ ? UseQueriesBuildersWithContractRouter<TRouter>
68
+ : never {
69
+ const path = options.path ?? []
70
+ const client = options.client as any
71
+
72
+ /**
73
+ * For sure root is not procedure so do not create builder on root
74
+ */
75
+ const builder = path.length ? createUseQueriesBuilder({ client, path }) : {}
76
+
77
+ return new Proxy(builder, {
78
+ get(target, key) {
79
+ const value = Reflect.get(target, key)
80
+
81
+ if (typeof key !== 'string') {
82
+ return value
83
+ }
84
+
85
+ const nextBuilders = createUseQueriesBuilders({
86
+ client: client[key],
87
+ path: [...path, key],
88
+ })
89
+
90
+ if (typeof value !== 'function') {
91
+ return nextBuilders
92
+ }
93
+
94
+ return new Proxy(value, {
95
+ get(_, key) {
96
+ return Reflect.get(nextBuilders, key)
97
+ },
98
+ })
99
+ },
100
+ }) as any
101
+ }
@@ -0,0 +1,64 @@
1
+ import type { SchemaOutput } from '@orpc/contract'
2
+ import {
3
+ ORPCContext,
4
+ type UserListOutputSchema,
5
+ type UserSchema,
6
+ orpcClient,
7
+ } from '../../tests/orpc'
8
+ import { createUseQueriesBuilders } from './builders'
9
+ import { useQueriesFactory } from './hook'
10
+
11
+ describe('useQueriesFactory', () => {
12
+ const useQueries = useQueriesFactory({
13
+ context: ORPCContext,
14
+ })
15
+
16
+ it('simple', () => {
17
+ const [findQuery, listQuery] = useQueries((o) => {
18
+ expectTypeOf(o).toEqualTypeOf(
19
+ createUseQueriesBuilders({ client: orpcClient }),
20
+ )
21
+
22
+ return [o.user.find({ id: '123' }), o.user.list({})]
23
+ })
24
+
25
+ expectTypeOf(findQuery.data).toEqualTypeOf<
26
+ SchemaOutput<typeof UserSchema> | undefined
27
+ >()
28
+
29
+ expectTypeOf(listQuery.data).toEqualTypeOf<
30
+ SchemaOutput<typeof UserListOutputSchema> | undefined
31
+ >()
32
+ })
33
+
34
+ it('with combine', () => {
35
+ const [findData, listData] = useQueries(
36
+ (o) => {
37
+ expectTypeOf(o).toEqualTypeOf(
38
+ createUseQueriesBuilders({ client: orpcClient }),
39
+ )
40
+
41
+ return [o.user.find({ id: '123' }), o.user.list({})]
42
+ },
43
+ ([findQuery, listQuery]) => {
44
+ expectTypeOf(findQuery.data).toEqualTypeOf<
45
+ SchemaOutput<typeof UserSchema> | undefined
46
+ >()
47
+
48
+ expectTypeOf(listQuery.data).toEqualTypeOf<
49
+ SchemaOutput<typeof UserListOutputSchema> | undefined
50
+ >()
51
+
52
+ return [findQuery.data, listQuery.data] as const
53
+ },
54
+ )
55
+
56
+ expectTypeOf(findData).toEqualTypeOf<
57
+ SchemaOutput<typeof UserSchema> | undefined
58
+ >()
59
+
60
+ expectTypeOf(listData).toEqualTypeOf<
61
+ SchemaOutput<typeof UserListOutputSchema> | undefined
62
+ >()
63
+ })
64
+ })
@@ -0,0 +1,89 @@
1
+ import { renderHook, waitFor } from '@testing-library/react'
2
+ import { ORPCContext, orpcClient, wrapper } from '../../tests/orpc'
3
+ import { createUseQueriesBuilders } from './builders'
4
+ import { useQueriesFactory } from './hook'
5
+
6
+ describe('useQueriesFactory', () => {
7
+ const useQueries = useQueriesFactory({
8
+ context: ORPCContext,
9
+ })
10
+
11
+ it('simple', async () => {
12
+ const queries = renderHook(
13
+ () =>
14
+ useQueries((o) => {
15
+ expectTypeOf(o).toEqualTypeOf(
16
+ createUseQueriesBuilders({ client: orpcClient }),
17
+ )
18
+
19
+ return [o.user.find({ id: '123' }), o.user.list({})]
20
+ }),
21
+ { wrapper: wrapper },
22
+ )
23
+
24
+ await waitFor(() =>
25
+ expect(queries.result.current[0].data).toEqual({
26
+ id: '123',
27
+ name: 'name-123',
28
+ }),
29
+ )
30
+
31
+ await waitFor(() =>
32
+ expect(queries.result.current[1].data).toEqual({
33
+ nextCursor: 2,
34
+ users: [
35
+ {
36
+ id: 'id-0',
37
+ name: 'number-0',
38
+ },
39
+ {
40
+ id: 'id-1',
41
+ name: 'number-1',
42
+ },
43
+ ],
44
+ }),
45
+ )
46
+ })
47
+
48
+ it('with combine', async () => {
49
+ const data = renderHook(
50
+ () =>
51
+ useQueries(
52
+ (o) => {
53
+ expectTypeOf(o).toEqualTypeOf(
54
+ createUseQueriesBuilders({ client: orpcClient }),
55
+ )
56
+
57
+ return [o.user.find({ id: '123' }), o.user.list({})]
58
+ },
59
+ ([findQuery, listQuery]) => {
60
+ return [findQuery.data, listQuery.data] as const
61
+ },
62
+ ),
63
+ { wrapper },
64
+ )
65
+
66
+ await waitFor(() =>
67
+ expect(data.result.current[0]).toEqual({
68
+ id: '123',
69
+ name: 'name-123',
70
+ }),
71
+ )
72
+
73
+ await waitFor(() =>
74
+ expect(data.result.current[1]).toEqual({
75
+ nextCursor: 2,
76
+ users: [
77
+ {
78
+ id: 'id-0',
79
+ name: 'number-0',
80
+ },
81
+ {
82
+ id: 'id-1',
83
+ name: 'number-1',
84
+ },
85
+ ],
86
+ }),
87
+ )
88
+ })
89
+ })
@@ -0,0 +1,57 @@
1
+ import type { ContractRouter } from '@orpc/contract'
2
+ import type { Router } from '@orpc/server'
3
+ import {
4
+ type QueriesOptions,
5
+ type QueriesResults,
6
+ useQueries,
7
+ } from '@tanstack/react-query'
8
+ import { type ORPCContext, useORPCContext } from '../react-context'
9
+ import {
10
+ type UseQueriesBuildersWithContractRouter,
11
+ type UseQueriesBuildersWithRouter,
12
+ createUseQueriesBuilders,
13
+ } from './builders'
14
+
15
+ export interface UseQueriesWithContractRouter<TRouter extends ContractRouter> {
16
+ <T extends Array<any> = [], TCombinedResult = QueriesResults<T>>(
17
+ build: (
18
+ builders: UseQueriesBuildersWithContractRouter<TRouter>,
19
+ ) => [...QueriesOptions<T>],
20
+ combine?: (result: QueriesResults<T>) => TCombinedResult,
21
+ ): TCombinedResult
22
+ }
23
+
24
+ export interface UseQueriesWithRouter<TRouter extends Router<any>> {
25
+ <T extends Array<any> = [], TCombinedResult = QueriesResults<T>>(
26
+ build: (
27
+ builders: UseQueriesBuildersWithRouter<TRouter>,
28
+ ) => [...QueriesOptions<T>],
29
+ combine?: (result: QueriesResults<T>) => TCombinedResult,
30
+ ): TCombinedResult
31
+ }
32
+
33
+ export interface UseQueriesFactoryOptions<
34
+ TRouter extends Router<any> | ContractRouter,
35
+ > {
36
+ context: ORPCContext<TRouter>
37
+ }
38
+
39
+ export function useQueriesFactory<TRouter extends Router<any> | ContractRouter>(
40
+ options: UseQueriesFactoryOptions<TRouter>,
41
+ ): TRouter extends Router<any>
42
+ ? UseQueriesWithRouter<TRouter>
43
+ : TRouter extends ContractRouter
44
+ ? UseQueriesWithContractRouter<TRouter>
45
+ : never {
46
+ const hook = (build: any, combine?: any) => {
47
+ const orpc = useORPCContext(options.context)
48
+ const builders = createUseQueriesBuilders({ client: orpc.client as any })
49
+
50
+ return useQueries({
51
+ queries: build(builders),
52
+ combine: combine,
53
+ })
54
+ }
55
+
56
+ return hook as any
57
+ }