diff --git a/examples/todo-app/src/resources/TodoResource.ts b/examples/todo-app/src/resources/TodoResource.ts index cd089155a0c8..d886d6cd2632 100644 --- a/examples/todo-app/src/resources/TodoResource.ts +++ b/examples/todo-app/src/resources/TodoResource.ts @@ -21,11 +21,6 @@ export const TodoResource = createPlaceholderResource({ }); export const queryRemainingTodos = new Query( - new schema.All(Todo), - (entries, { userId } = {}) => { - if (userId !== undefined) - return entries.filter((todo) => todo.userId === userId && !todo.completed) - .length; - return entries.filter((todo) => !todo.completed).length; - }, + TodoResource.getList.schema, + (entries) => entries && entries.filter((todo) => !todo.completed).length, ); diff --git a/packages/endpoint/src/index.ts b/packages/endpoint/src/index.ts index 711b1d9eeef7..83d753829e8e 100644 --- a/packages/endpoint/src/index.ts +++ b/packages/endpoint/src/index.ts @@ -34,6 +34,7 @@ export type { NormalizeNullable, Denormalize, DenormalizeNullable, + SchemaToArgs, } from './normal.js'; export type { EndpointExtraOptions, diff --git a/packages/endpoint/src/interface.ts b/packages/endpoint/src/interface.ts index 8b4faf1f647e..92fb117f4bc7 100644 --- a/packages/endpoint/src/interface.ts +++ b/packages/endpoint/src/interface.ts @@ -14,7 +14,7 @@ export type Serializable< T extends { toJSON(): string } = { toJSON(): string }, > = (value: any) => T; -export interface SchemaSimple { +export interface SchemaSimple { normalize( input: any, parent: any, @@ -23,7 +23,7 @@ export interface SchemaSimple { addEntity: (...args: any) => any, visitedEntities: Record, storeEntities: any, - args?: any[], + args: Args, ): any; denormalize( input: {}, diff --git a/packages/endpoint/src/normal.ts b/packages/endpoint/src/normal.ts index 9ba85d72c7b2..45d6d4952809 100644 --- a/packages/endpoint/src/normal.ts +++ b/packages/endpoint/src/normal.ts @@ -131,3 +131,31 @@ export type NormalizedSchema = { export interface EntityMap { readonly [k: string]: EntityInterface; } + +export type SchemaToArgs< + S extends { + normalize( + input: any, + parent: any, + key: any, + visit: (...args: any) => any, + addEntity: (...args: any) => any, + visitedEntities: Record, + storeEntities: any, + args: any, + ): any; + }, +> = S extends { + normalize( + input: any, + parent: any, + key: any, + visit: (...args: any) => any, + addEntity: (...args: any) => any, + visitedEntities: Record, + storeEntities: any, + args: infer Args, + ): any; +} + ? Args + : never; diff --git a/packages/endpoint/src/queryEndpoint.ts b/packages/endpoint/src/queryEndpoint.ts index dddd277a588a..b422b0dfedae 100644 --- a/packages/endpoint/src/queryEndpoint.ts +++ b/packages/endpoint/src/queryEndpoint.ts @@ -3,7 +3,7 @@ import type { NormalizedIndex, SchemaSimple, } from './interface.js'; -import type { Denormalize } from './normal.js'; +import type { Denormalize, SchemaToArgs } from './normal.js'; /** * Programmatic cache reading @@ -11,7 +11,7 @@ import type { Denormalize } from './normal.js'; */ export class Query< S extends SchemaSimple, - P extends any[] = [], + P extends SchemaToArgs = SchemaToArgs, R = Denormalize, > { declare schema: QuerySchema; @@ -34,17 +34,15 @@ export class Query< protected createQuerySchema(schema: SchemaSimple) { const query = Object.create(schema); - query.denormalize = ( - { args, input }: { args: P; input: any }, - _: P, - unvisit: any, - ) => { + query.denormalize = (input: any, args: P, unvisit: any) => { if (input === undefined) return undefined; - const value = (schema as any).denormalize(input, args, unvisit); + const value = unvisit(input, schema); return typeof value === 'symbol' ? undefined : this.process(value, ...args); }; + // anywhere in the hierarchy + if ('key' in schema) query.key = `QUERY ${schema.key}`; query.infer = ( args: any, indexes: any, @@ -56,7 +54,7 @@ export class Query< ) => any, entities: EntityTable, ) => { - return { args, input: recurse(schema, args, indexes, entities) }; + return recurse(schema, args, indexes, entities); }; return query; } diff --git a/packages/endpoint/src/schema.d.ts b/packages/endpoint/src/schema.d.ts index 614f29ccf3c8..7219bb248dae 100644 --- a/packages/endpoint/src/schema.d.ts +++ b/packages/endpoint/src/schema.d.ts @@ -317,11 +317,12 @@ export declare let CollectionRoot: CollectionConstructor; */ export declare class Collection< S extends any[] | PolymorphicInterface = any, - Parent extends any[] = [ + Args extends any[] = [ urlParams: Record, body?: Record, ], -> extends CollectionRoot {} + Parent = any, +> extends CollectionRoot {} // id is in Instance, so we default to that as pk /** diff --git a/packages/endpoint/src/schemaTypes.ts b/packages/endpoint/src/schemaTypes.ts index 72c439691696..049655763291 100644 --- a/packages/endpoint/src/schemaTypes.ts +++ b/packages/endpoint/src/schemaTypes.ts @@ -24,9 +24,10 @@ export type CollectionArrayAdder = S extends { export interface CollectionInterface< S extends PolymorphicInterface = any, - Parent extends any[] = any, + Args extends any[] = any[], + Parent = any, > { - addWith

( + addWith

( merge: (existing: any, incoming: any) => any, createCollectionFilter?: ( ...args: P @@ -46,7 +47,7 @@ export interface CollectionInterface< addEntity: (...args: any) => any, visitedEntities: Record, storeEntities: any, - args: any[], + args: Args, ): string; merge(existing: any, incoming: any): any; @@ -126,28 +127,34 @@ export interface CollectionInterface< * @see https://dataclient.io/rest/api/Collection#assign */ assign: S extends { denormalize(...args: any): Record } - ? schema.Collection + ? schema.Collection : never; } export type CollectionFromSchema< S extends any[] | PolymorphicInterface = any, - Parent extends any[] = [ + Args extends any[] = [ urlParams: Record, body?: Record, ], -> = CollectionInterface : S, Parent>; + Parent = any, +> = CollectionInterface< + S extends any[] ? schema.Array : S, + Args, + Parent +>; export interface CollectionConstructor { new < S extends SchemaSimple[] | PolymorphicInterface = any, - Parent extends any[] = [ + Args extends any[] = [ urlParams: Record, body?: Record, ], + Parent = any, >( schema: S, - options?: CollectionOptions, - ): CollectionFromSchema; + options?: CollectionOptions, + ): CollectionFromSchema; readonly prototype: CollectionInterface; } diff --git a/packages/endpoint/src/schemas/Collection.ts b/packages/endpoint/src/schemas/Collection.ts index 55132cc05626..1509921c1ef8 100644 --- a/packages/endpoint/src/schemas/Collection.ts +++ b/packages/endpoint/src/schemas/Collection.ts @@ -21,10 +21,11 @@ const createValue = (value: any) => ({ ...value }); */ export default class CollectionSchema< S extends PolymorphicInterface = any, - Parent extends any[] = [ + Args extends any[] = [ urlParams: Record, body?: Record, ], + Parent = any, > { protected declare nestKey: (parent: any, key: string) => Record; @@ -35,18 +36,18 @@ export default class CollectionSchema< declare readonly key: string; declare push: S extends ArraySchema - ? CollectionSchema + ? CollectionSchema : undefined; declare unshift: S extends ArraySchema - ? CollectionSchema + ? CollectionSchema : undefined; declare assign: S extends Values - ? CollectionSchema + ? CollectionSchema : undefined; - addWith

( + addWith

( merge: (existing: any, incoming: any) => any, createCollectionFilter?: ( ...args: P @@ -59,7 +60,7 @@ export default class CollectionSchema< // so fetch(create, { userId: 'bob', completed: true }, data) // would possibly add to {}, {userId: 'bob'}, {completed: true}, {userId: 'bob', completed: true } - but only those already in the store // it ignores keys that start with sort as those are presumed to not filter results - protected createCollectionFilter(...args: Parent) { + protected createCollectionFilter(...args: Args) { return (collectionKey: Record) => Object.entries(collectionKey).every( ([key, value]) => @@ -74,7 +75,7 @@ export default class CollectionSchema< return key.startsWith('order'); } - constructor(schema: S, options?: CollectionOptions) { + constructor(schema: S, options?: CollectionOptions) { this.schema = Array.isArray(schema) ? (new ArraySchema(schema[0]) as any) : schema; @@ -109,7 +110,7 @@ export default class CollectionSchema< this.createCollectionFilter = ( options as any as { createCollectionFilter: ( - ...args: Parent + ...args: Args ) => (collectionKey: Record) => boolean; } ).createCollectionFilter.bind(this) as any; @@ -152,13 +153,13 @@ export default class CollectionSchema< normalize( input: any, - parent: any, + parent: Parent, key: string, visit: (...args: any) => any, addEntity: (...args: any) => any, visitedEntities: Record, storeEntities: any, - args: any[], + args: Args, ): string { const pkList = this.schema.normalize( input, @@ -239,22 +240,23 @@ export default class CollectionSchema< } export type CollectionOptions< - Parent extends any[] = [ + Args extends any[] = [ urlParams: Record, body?: Record, ], + Parent = any, > = ( | { - nestKey?: (parent: any, key: string) => Record; + nestKey?: (parent: Parent, key: string) => Record; } | { - argsKey?: (...args: any) => Record; + argsKey?: (...args: Args) => Record; } ) & ( | { createCollectionFilter?: ( - ...args: Parent + ...args: Args ) => (collectionKey: Record) => boolean; } | { diff --git a/packages/endpoint/src/schemas/__tests__/Collection.test.ts b/packages/endpoint/src/schemas/__tests__/Collection.test.ts index f3e964683461..53ca84309b93 100644 --- a/packages/endpoint/src/schemas/__tests__/Collection.test.ts +++ b/packages/endpoint/src/schemas/__tests__/Collection.test.ts @@ -88,6 +88,7 @@ describe(`${schema.Collection.name} normalization`, () => { () => undefined, {}, {}, + // @ts-expect-error [], ); } diff --git a/packages/endpoint/src/schemas/__tests__/Query.test.ts b/packages/endpoint/src/schemas/__tests__/Query.test.ts index 2c5b7677ede2..f92818c1d310 100644 --- a/packages/endpoint/src/schemas/__tests__/Query.test.ts +++ b/packages/endpoint/src/schemas/__tests__/Query.test.ts @@ -80,6 +80,9 @@ describe.each([ inferResults(sortedUsers.schema, [{ asc: true }], {}, entities), sortedUsers.schema, createInput(entities), + {}, + new WeakEntityMap(), + [{ asc: true }], ), ).toMatchSnapshot(); }); @@ -137,6 +140,9 @@ describe.each([ ), userCountByAdmin.schema, createInput(entities), + {}, + new WeakEntityMap(), + [{ isAdmin: false }], ); expect(nonAdminCount).toBe(3); const adminCount: @@ -150,6 +156,9 @@ describe.each([ ), userCountByAdmin.schema, createInput(entities), + {}, + new WeakEntityMap(), + [{ isAdmin: true }], ); expect(adminCount).toBe(1); if (typeof totalCount === 'symbol') return; diff --git a/packages/rest/src/resourceTypes.ts b/packages/rest/src/resourceTypes.ts index 03399c73e386..8847fb873295 100644 --- a/packages/rest/src/resourceTypes.ts +++ b/packages/rest/src/resourceTypes.ts @@ -72,7 +72,16 @@ export interface Resource< ? GetEndpoint< { path: ShortenPath; - schema: schema.Collection<[O['schema']]>; + schema: schema.Collection< + [O['schema']], + [ + 'searchParams' extends keyof O + ? O['searchParams'] extends undefined + ? PathArgs> + : O['searchParams'] & PathArgs> + : PathArgs>, + ] + >; body: 'body' extends keyof O ? O['body'] : Partial>; @@ -82,7 +91,16 @@ export interface Resource< : GetEndpoint< { path: ShortenPath; - schema: schema.Collection<[O['schema']]>; + schema: schema.Collection< + [O['schema']], + [ + 'searchParams' extends keyof O + ? O['searchParams'] extends undefined + ? PathArgs> + : O['searchParams'] & PathArgs> + : PathArgs>, + ] + >; body: 'body' extends keyof O ? O['body'] : Partial>;