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 21, 2023
1 parent f93fc16 commit c9d3179
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 c9d3179

@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: c9d3179 Previous: 43b2e41 Ratio
normalizeLong 314 ops/sec (±1.97%) 266 ops/sec (±0.69%) 0.85
infer All 6973 ops/sec (±0.30%) 5526 ops/sec (±0.39%) 0.79
denormalizeLong 205 ops/sec (±0.64%) 171 ops/sec (±0.86%) 0.83
denormalizeLong donotcache 547 ops/sec (±0.16%) 458 ops/sec (±0.21%) 0.84
denormalizeShort donotcache 500x 842 ops/sec (±1.30%) 804 ops/sec (±1.65%) 0.95
denormalizeShort 500x 563 ops/sec (±2.05%) 579 ops/sec (±0.25%) 1.03
denormalizeLong with mixin Entity 199 ops/sec (±0.59%) 170 ops/sec (±0.78%) 0.85
denormalizeLong withCache 4363 ops/sec (±0.05%) 3771 ops/sec (±0.14%) 0.86
denormalizeLongAndShort withEntityCacheOnly 1018 ops/sec (±0.44%) 849 ops/sec (±0.33%) 0.83
denormalizeLong All withCache 4806 ops/sec (±0.22%) 4329 ops/sec (±0.10%) 0.90
denormalizeLong Query-sorted withCache 4975 ops/sec (±0.28%) 4344 ops/sec (±0.23%) 0.87
getResponse 4347 ops/sec (±3.32%) 3312 ops/sec (±2.27%) 0.76
getResponse (null) 2392871 ops/sec (±0.10%) 2258196 ops/sec (±0.15%) 0.94
getResponse (clear cache) 172 ops/sec (±0.60%) 160 ops/sec (±0.98%) 0.93
getSmallResponse 1717 ops/sec (±0.14%) 1667 ops/sec (±0.10%) 0.97
getSmallInferredResponse 1261 ops/sec (±0.19%) 1427 ops/sec (±0.16%) 1.13
getResponse Query-sorted 484 ops/sec (±0.75%) 397 ops/sec (±1.97%) 0.82
setLong 310 ops/sec (±2.71%) 265 ops/sec (±0.68%) 0.85
setLongWithMerge 127 ops/sec (±0.23%) 117 ops/sec (±0.46%) 0.92
setLongWithSimpleMerge 139 ops/sec (±0.26%) 126 ops/sec (±0.54%) 0.91

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

Please sign in to comment.