Skip to content

Commit

Permalink
update query work
Browse files Browse the repository at this point in the history
  • Loading branch information
ntucker committed Oct 23, 2023
1 parent 000e19f commit c0f5c92
Show file tree
Hide file tree
Showing 12 changed files with 433 additions and 171 deletions.
9 changes: 2 additions & 7 deletions examples/todo-app/src/resources/TodoResource.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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,
);
1 change: 1 addition & 0 deletions packages/endpoint/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ export type {
NormalizeNullable,
Denormalize,
DenormalizeNullable,
SchemaToArgs,
} from './normal.js';
export type {
EndpointExtraOptions,
Expand Down
4 changes: 2 additions & 2 deletions packages/endpoint/src/interface.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ export type Serializable<
T extends { toJSON(): string } = { toJSON(): string },
> = (value: any) => T;

export interface SchemaSimple<T = any> {
export interface SchemaSimple<T = any, Args extends any[] = any[]> {
normalize(
input: any,
parent: any,
Expand All @@ -23,7 +23,7 @@ export interface SchemaSimple<T = any> {
addEntity: (...args: any) => any,
visitedEntities: Record<string, any>,
storeEntities: any,
args?: any[],
args: Args,
): any;
denormalize(
input: {},
Expand Down
28 changes: 28 additions & 0 deletions packages/endpoint/src/normal.ts
Original file line number Diff line number Diff line change
Expand Up @@ -131,3 +131,31 @@ export type NormalizedSchema<E, R> = {
export interface EntityMap<T = any> {
readonly [k: string]: EntityInterface<T>;
}

export type SchemaToArgs<
S extends {
normalize(
input: any,
parent: any,
key: any,
visit: (...args: any) => any,
addEntity: (...args: any) => any,
visitedEntities: Record<string, any>,
storeEntities: any,
args: any,
): any;
},
> = S extends {
normalize(
input: any,
parent: any,
key: any,
visit: (...args: any) => any,
addEntity: (...args: any) => any,
visitedEntities: Record<string, any>,
storeEntities: any,
args: infer Args,
): any;
}
? Args
: never;
16 changes: 7 additions & 9 deletions packages/endpoint/src/queryEndpoint.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,15 +3,15 @@ import type {
NormalizedIndex,
SchemaSimple,
} from './interface.js';
import type { Denormalize } from './normal.js';
import type { Denormalize, SchemaToArgs } from './normal.js';

/**
* Programmatic cache reading
* @see https://dataclient.io/rest/api/Query
*/
export class Query<
S extends SchemaSimple,
P extends any[] = [],
P extends SchemaToArgs<S> = SchemaToArgs<S>,
R = Denormalize<S>,
> {
declare schema: QuerySchema<S, R>;
Expand All @@ -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,
Expand All @@ -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;
}
Expand Down
5 changes: 3 additions & 2 deletions packages/endpoint/src/schema.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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<string, any>,
body?: Record<string, any>,
],
> extends CollectionRoot<S, Parent> {}
Parent = any,
> extends CollectionRoot<S, Args, Parent> {}

// id is in Instance, so we default to that as pk
/**
Expand Down
25 changes: 16 additions & 9 deletions packages/endpoint/src/schemaTypes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,9 +24,10 @@ export type CollectionArrayAdder<S extends PolymorphicInterface> = S extends {

export interface CollectionInterface<
S extends PolymorphicInterface = any,
Parent extends any[] = any,
Args extends any[] = any[],
Parent = any,
> {
addWith<P extends any[] = Parent>(
addWith<P extends any[] = Args>(
merge: (existing: any, incoming: any) => any,
createCollectionFilter?: (
...args: P
Expand All @@ -46,7 +47,7 @@ export interface CollectionInterface<
addEntity: (...args: any) => any,
visitedEntities: Record<string, any>,
storeEntities: any,
args: any[],
args: Args,
): string;

merge(existing: any, incoming: any): any;
Expand Down Expand Up @@ -126,28 +127,34 @@ export interface CollectionInterface<
* @see https://dataclient.io/rest/api/Collection#assign
*/
assign: S extends { denormalize(...args: any): Record<string, unknown> }
? schema.Collection<S, Parent>
? schema.Collection<S, Args, Parent>
: never;
}
export type CollectionFromSchema<
S extends any[] | PolymorphicInterface = any,
Parent extends any[] = [
Args extends any[] = [
urlParams: Record<string, any>,
body?: Record<string, any>,
],
> = CollectionInterface<S extends any[] ? schema.Array<S[number]> : S, Parent>;
Parent = any,
> = CollectionInterface<
S extends any[] ? schema.Array<S[number]> : S,
Args,
Parent
>;

export interface CollectionConstructor {
new <
S extends SchemaSimple[] | PolymorphicInterface = any,
Parent extends any[] = [
Args extends any[] = [
urlParams: Record<string, any>,
body?: Record<string, any>,
],
Parent = any,
>(
schema: S,
options?: CollectionOptions,
): CollectionFromSchema<S, Parent>;
options?: CollectionOptions<Args, Parent>,
): CollectionFromSchema<S, Args, Parent>;
readonly prototype: CollectionInterface;
}

Expand Down
30 changes: 16 additions & 14 deletions packages/endpoint/src/schemas/Collection.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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<string, any>,
body?: Record<string, any>,
],
Parent = any,
> {
protected declare nestKey: (parent: any, key: string) => Record<string, any>;

Expand All @@ -35,18 +36,18 @@ export default class CollectionSchema<
declare readonly key: string;

declare push: S extends ArraySchema<any>
? CollectionSchema<S, Parent>
? CollectionSchema<S, Args, Parent>
: undefined;

declare unshift: S extends ArraySchema<any>
? CollectionSchema<S, Parent>
? CollectionSchema<S, Args, Parent>
: undefined;

declare assign: S extends Values<any>
? CollectionSchema<S, Parent>
? CollectionSchema<S, Args, Parent>
: undefined;

addWith<P extends any[] = Parent>(
addWith<P extends any[] = Args>(
merge: (existing: any, incoming: any) => any,
createCollectionFilter?: (
...args: P
Expand All @@ -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<string, string>) =>
Object.entries(collectionKey).every(
([key, value]) =>
Expand All @@ -74,7 +75,7 @@ export default class CollectionSchema<
return key.startsWith('order');
}

constructor(schema: S, options?: CollectionOptions) {
constructor(schema: S, options?: CollectionOptions<Args, Parent>) {
this.schema = Array.isArray(schema)
? (new ArraySchema(schema[0]) as any)
: schema;
Expand Down Expand Up @@ -109,7 +110,7 @@ export default class CollectionSchema<
this.createCollectionFilter = (
options as any as {
createCollectionFilter: (
...args: Parent
...args: Args
) => (collectionKey: Record<string, string>) => boolean;
}
).createCollectionFilter.bind(this) as any;
Expand Down Expand Up @@ -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<string, any>,
storeEntities: any,
args: any[],
args: Args,
): string {
const pkList = this.schema.normalize(
input,
Expand Down Expand Up @@ -239,22 +240,23 @@ export default class CollectionSchema<
}

export type CollectionOptions<
Parent extends any[] = [
Args extends any[] = [
urlParams: Record<string, any>,
body?: Record<string, any>,
],
Parent = any,
> = (
| {
nestKey?: (parent: any, key: string) => Record<string, any>;
nestKey?: (parent: Parent, key: string) => Record<string, any>;
}
| {
argsKey?: (...args: any) => Record<string, any>;
argsKey?: (...args: Args) => Record<string, any>;
}
) &
(
| {
createCollectionFilter?: (
...args: Parent
...args: Args
) => (collectionKey: Record<string, string>) => boolean;
}
| {
Expand Down
1 change: 1 addition & 0 deletions packages/endpoint/src/schemas/__tests__/Collection.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ describe(`${schema.Collection.name} normalization`, () => {
() => undefined,
{},
{},
// @ts-expect-error
[],
);
}
Expand Down
Loading

1 comment on commit c0f5c92

@github-actions
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Benchmark

Benchmark suite Current: c0f5c92 Previous: 43b2e41 Ratio
normalizeLong 220 ops/sec (±2.85%) 266 ops/sec (±0.69%) 1.21
infer All 4859 ops/sec (±1.49%) 5526 ops/sec (±0.39%) 1.14
denormalizeLong 152 ops/sec (±1.79%) 171 ops/sec (±0.86%) 1.13
denormalizeLong donotcache 411 ops/sec (±1.58%) 458 ops/sec (±0.21%) 1.11
denormalizeShort donotcache 500x 704 ops/sec (±4.99%) 804 ops/sec (±1.65%) 1.14
denormalizeShort 500x 522 ops/sec (±2.47%) 579 ops/sec (±0.25%) 1.11
denormalizeLong with mixin Entity 149 ops/sec (±1.84%) 170 ops/sec (±0.78%) 1.14
denormalizeLong withCache 3407 ops/sec (±1.13%) 3771 ops/sec (±0.14%) 1.11
denormalizeLongAndShort withEntityCacheOnly 688 ops/sec (±1.62%) 849 ops/sec (±0.33%) 1.23
denormalizeLong All withCache 3542 ops/sec (±1.27%) 4329 ops/sec (±0.10%) 1.22
denormalizeLong Query-sorted withCache 3604 ops/sec (±1.30%) 4344 ops/sec (±0.23%) 1.21
getResponse 2834 ops/sec (±2.07%) 3312 ops/sec (±2.27%) 1.17
getResponse (null) 2090495 ops/sec (±1.10%) 2258196 ops/sec (±0.15%) 1.08
getResponse (clear cache) 140 ops/sec (±1.56%) 160 ops/sec (±0.98%) 1.14
getSmallResponse 1322 ops/sec (±1.29%) 1667 ops/sec (±0.10%) 1.26
getSmallInferredResponse 1134 ops/sec (±1.26%) 1427 ops/sec (±0.16%) 1.26
getResponse Query-sorted 317 ops/sec (±2.12%) 397 ops/sec (±1.97%) 1.25
setLong 212 ops/sec (±3.49%) 265 ops/sec (±0.68%) 1.25
setLongWithMerge 96.09 ops/sec (±1.61%) 117 ops/sec (±0.46%) 1.22
setLongWithSimpleMerge 110 ops/sec (±1.87%) 126 ops/sec (±0.54%) 1.15

This comment was automatically generated by workflow using github-action-benchmark.

Please sign in to comment.