Skip to content

Commit b51e0c8

Browse files
committed
feat(serve): add interface of data source, pagination and pass to data query builder
- add pagination field in "APISchema". - add "PaginationStrategy" interface and offset, cursor, keyset based strategy class. - add "PaginationTransformer" to do transfrom from koa ctx . - add data source inerface "IDataSource". - refactor "BaseRoute" for splting request and pagination transformer to implmented restful & graphql class, because of graphql get request from body. - update test cases of each sql clause method of "DataQueryBuilder".
1 parent 7bf9ed3 commit b51e0c8

29 files changed

+672
-200
lines changed

packages/core/src/models/artifact.ts

+15
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,12 @@ error:
1818
message: 'You are not allowed to access this resource'
1919
*/
2020

21+
export enum PaginationMode {
22+
CURSOR = 'CURSOR',
23+
OFFSET = 'OFFSET',
24+
KEYSET = 'KEYSET',
25+
}
26+
2127
export enum FieldInType {
2228
QUERY = 'QUERY',
2329
HEADER = 'HEADER',
@@ -44,6 +50,12 @@ export interface RequestSchema {
4450
validators: Array<ValidatorDefinition>;
4551
}
4652

53+
export interface PaginationSchema {
54+
mode: PaginationMode;
55+
// The key name used for do filtering by key for keyset pagination.
56+
keyName?: string;
57+
}
58+
4759
export interface ErrorInfo {
4860
code: string;
4961
message: string;
@@ -59,6 +71,9 @@ export interface APISchema {
5971
request: Array<RequestSchema>;
6072
errors: Array<ErrorInfo>;
6173
response: any;
74+
// The pagination strategy that do paginate when querying
75+
// If not set pagination, then API request not provide the field to do it
76+
pagination?: PaginationSchema;
6277
}
6378

6479
export interface BuiltArtifact {

packages/serve/src/lib/data-query/builder/dataQueryBuilder.ts

+42-7
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
import { IDataSource } from '@data-source/.';
2+
import { Pagination } from '@route/.';
3+
14
import { find, isEmpty } from 'lodash';
25
import {
36
ComparisonPredicate,
@@ -177,6 +180,8 @@ export interface SQLClauseOperation {
177180
export interface IDataQueryBuilder {
178181
readonly statement: string;
179182
readonly operations: SQLClauseOperation;
183+
readonly dataSource: IDataSource;
184+
180185
// Select clause methods
181186
select(...columns: Array<SelectedColumn | string>): DataQueryBuilder;
182187
distinct(...columns: Array<SelectedColumn | string>): DataQueryBuilder;
@@ -386,21 +391,29 @@ export interface IDataQueryBuilder {
386391
limit(size: number): DataQueryBuilder;
387392
offset(move: number): DataQueryBuilder;
388393
take(size: number, move: number): DataQueryBuilder;
394+
// paginate
395+
paginate(pagination: Pagination): void;
396+
value(): Promise<object>;
397+
clone(): IDataQueryBuilder;
389398
}
390399

391400
export class DataQueryBuilder implements IDataQueryBuilder {
392401
public readonly statement: string;
393402
// record all operations for different SQL clauses
394403
public readonly operations: SQLClauseOperation;
395-
404+
public readonly dataSource: IDataSource;
405+
public pagination?: Pagination;
396406
constructor({
397407
statement,
398408
operations,
409+
dataSource,
399410
}: {
400411
statement: string;
401412
operations?: SQLClauseOperation;
413+
dataSource: IDataSource;
402414
}) {
403415
this.statement = statement;
416+
this.dataSource = dataSource;
404417
this.operations = operations || {
405418
select: null,
406419
where: [],
@@ -601,6 +614,7 @@ export class DataQueryBuilder implements IDataQueryBuilder {
601614
public whereWrapped(builderCallback: BuilderClauseCallback) {
602615
const wrappedBuilder = new DataQueryBuilder({
603616
statement: '',
617+
dataSource: this.dataSource,
604618
});
605619
builderCallback(wrappedBuilder);
606620
this.recordWhere({
@@ -1043,6 +1057,33 @@ export class DataQueryBuilder implements IDataQueryBuilder {
10431057
return this;
10441058
}
10451059

1060+
public clone() {
1061+
return new DataQueryBuilder({
1062+
statement: this.statement,
1063+
dataSource: this.dataSource,
1064+
operations: this.operations,
1065+
});
1066+
}
1067+
1068+
// setup pagination if would like to do paginate
1069+
public paginate(pagination: Pagination) {
1070+
this.pagination = pagination;
1071+
}
1072+
1073+
public async value() {
1074+
// call data source
1075+
const result = await this.dataSource.execute({
1076+
statement: this.statement,
1077+
operations: this.operations,
1078+
pagination: this.pagination,
1079+
});
1080+
1081+
// Reset operations
1082+
await this.resetOperations();
1083+
1084+
return result;
1085+
}
1086+
10461087
// record Select-On related operations
10471088
private recordSelect({
10481089
command,
@@ -1128,10 +1169,4 @@ export class DataQueryBuilder implements IDataQueryBuilder {
11281169
this.operations.limit = null;
11291170
this.operations.offset = null;
11301171
}
1131-
public value() {
1132-
// TODO: call Driver
1133-
1134-
// Reset operations
1135-
this.resetOperations();
1136-
}
11371172
}
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,15 @@
1+
import { IDataSource } from '@data-source/.';
12
import {
23
DataQueryBuilder,
34
IDataQueryBuilder,
45
} from './builder/dataQueryBuilder';
56

6-
export const dataQuery = (sqlStatement: string): IDataQueryBuilder => {
7+
export const dataQuery = (
8+
sqlStatement: string,
9+
dataSource: IDataSource
10+
): IDataQueryBuilder => {
711
return new DataQueryBuilder({
812
statement: sqlStatement,
13+
dataSource: dataSource,
914
});
1015
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
import { SQLClauseOperation } from '@data-query/.';
2+
import { Pagination } from '@route/.';
3+
4+
export interface IDataSource {
5+
execute({
6+
statement,
7+
operations,
8+
pagination,
9+
}: {
10+
statement: string;
11+
operations: SQLClauseOperation;
12+
pagination?: Pagination;
13+
}): Promise<object>;
14+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './dataSource';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './strategy';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import { KoaRouterContext } from '@route/route-component';
2+
import { normalizeStringValue, PaginationMode } from '@vulcan/core';
3+
import { PaginationStrategy } from './strategy';
4+
5+
export interface CursorPagination {
6+
limit: number;
7+
cursor: string;
8+
}
9+
10+
export class CursorBasedStrategy extends PaginationStrategy<CursorPagination> {
11+
public async transform(ctx: KoaRouterContext) {
12+
const checkFelidInHeader = ['limit', 'cursor'].every((field) =>
13+
Object.keys(ctx.request.query).includes(field)
14+
);
15+
if (!checkFelidInHeader)
16+
throw new Error(
17+
`The ${PaginationMode.CURSOR} must provide limit and cursor in query string.`
18+
);
19+
const limitVal = ctx.request.query['limit'] as string;
20+
const cursorVal = ctx.request.query['cursor'] as string;
21+
return {
22+
limit: normalizeStringValue(limitVal, 'limit', Number.name),
23+
cursor: normalizeStringValue(cursorVal, 'cursor', Number.name),
24+
} as CursorPagination;
25+
}
26+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
export * from './strategy';
2+
export * from './offsetBasedStrategy';
3+
export * from './cursorBasedStrategy';
4+
export * from './keysetBasedStrategy';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import {
2+
APISchema,
3+
normalizeStringValue,
4+
PaginationMode,
5+
PaginationSchema,
6+
} from '@vulcan/core';
7+
import { KoaRouterContext } from '@route/route-component';
8+
import { PaginationStrategy } from './strategy';
9+
10+
export interface KeysetPagination {
11+
limit: number;
12+
[keyName: string]: string | number;
13+
}
14+
15+
export class KeysetBasedStrategy extends PaginationStrategy<KeysetPagination> {
16+
private pagination: PaginationSchema;
17+
constructor(pagination: PaginationSchema) {
18+
super();
19+
this.pagination = pagination;
20+
}
21+
public async transform(ctx: KoaRouterContext) {
22+
if (!this.pagination.keyName)
23+
throw new Error(
24+
`The keyset pagination need to set "keyName" in schema for indicate what key need to do filter.`
25+
);
26+
const { keyName } = this.pagination;
27+
const checkFelidInHeader = ['limit', keyName].every((field) =>
28+
Object.keys(ctx.request.query).includes(field)
29+
);
30+
if (!checkFelidInHeader)
31+
throw new Error(
32+
`The ${PaginationMode.KEYSET} must provide limit and offset in query string.`
33+
);
34+
const limitVal = ctx.request.query['limit'] as string;
35+
const keyNameVal = ctx.request.query[keyName] as string;
36+
return {
37+
limit: normalizeStringValue(limitVal, 'limit', Number.name),
38+
[keyName]: normalizeStringValue(keyNameVal, keyName, Number.name),
39+
} as KeysetPagination;
40+
}
41+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import { normalizeStringValue, PaginationMode } from '@vulcan/core';
2+
import { KoaRouterContext } from '@route/route-component';
3+
import { PaginationStrategy } from './strategy';
4+
5+
export interface OffsetPagination {
6+
limit: number;
7+
offset: number;
8+
}
9+
10+
export class OffsetBasedStrategy extends PaginationStrategy<OffsetPagination> {
11+
public async transform(ctx: KoaRouterContext) {
12+
const checkFelidInHeader = ['limit', 'offset'].every((field) =>
13+
Object.keys(ctx.request.query).includes(field)
14+
);
15+
if (!checkFelidInHeader)
16+
throw new Error(
17+
`The ${PaginationMode.OFFSET} must provide limit and offset in query string.`
18+
);
19+
const limitVal = ctx.request.query['limit'] as string;
20+
const offsetVal = ctx.request.query['offset'] as string;
21+
return {
22+
limit: normalizeStringValue(limitVal, 'limit', Number.name),
23+
offset: normalizeStringValue(offsetVal, 'offset', Number.name),
24+
} as OffsetPagination;
25+
}
26+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
import { KoaRouterContext } from '@route/route-component';
2+
3+
export abstract class PaginationStrategy<T> {
4+
public abstract transform(ctx: KoaRouterContext): Promise<T>;
5+
}

packages/serve/src/lib/route/route-component/baseRoute.ts

+19-14
Original file line numberDiff line numberDiff line change
@@ -3,47 +3,52 @@ import { RouterContext as KoaRouterContext } from 'koa-router';
33
import { IRequestTransformer, RequestParameters } from './requestTransformer';
44
import { IRequestValidator } from './requestValidator';
55
import { APISchema, TemplateEngine } from '@vulcan/core';
6+
import { IPaginationTransformer, Pagination } from './paginationTransformer';
67
export { KoaRouterContext, KoaNext };
78

9+
interface TransformedRequest {
10+
reqParams: RequestParameters;
11+
pagination?: Pagination;
12+
}
13+
814
interface IRoute {
915
respond(ctx: KoaRouterContext): Promise<any>;
1016
}
1117

1218
export abstract class BaseRoute implements IRoute {
1319
public readonly apiSchema: APISchema;
14-
private readonly reqTransformer: IRequestTransformer;
15-
private readonly reqValidator: IRequestValidator;
16-
private readonly templateEngine: TemplateEngine;
17-
20+
protected readonly reqTransformer: IRequestTransformer;
21+
protected readonly reqValidator: IRequestValidator;
22+
protected readonly templateEngine: TemplateEngine;
23+
protected readonly paginationTransformer: IPaginationTransformer;
1824
constructor({
1925
apiSchema,
2026
reqTransformer,
2127
reqValidator,
28+
paginationTransformer,
2229
templateEngine,
2330
}: {
2431
apiSchema: APISchema;
2532
reqTransformer: IRequestTransformer;
2633
reqValidator: IRequestValidator;
34+
paginationTransformer: IPaginationTransformer;
2735
templateEngine: TemplateEngine;
2836
}) {
2937
this.apiSchema = apiSchema;
3038
this.reqTransformer = reqTransformer;
3139
this.reqValidator = reqValidator;
40+
this.paginationTransformer = paginationTransformer;
3241
this.templateEngine = templateEngine;
3342
}
3443

35-
public async respond(ctx: KoaRouterContext) {
36-
const params = await this.reqTransformer.transform(ctx, this.apiSchema);
37-
await this.reqValidator.validate(params, this.apiSchema);
38-
return await this.handleRequest(ctx, params);
39-
}
44+
public abstract respond(ctx: KoaRouterContext): Promise<any>;
4045

41-
protected abstract handleRequest(
42-
ctx: KoaRouterContext,
43-
reqParams: RequestParameters
44-
): Promise<any>;
46+
protected abstract prepare(
47+
ctx: KoaRouterContext
48+
): Promise<TransformedRequest>;
4549

46-
protected async runQuery(reqParams: RequestParameters) {
50+
protected async handle(transformed: TransformedRequest) {
51+
const { reqParams } = transformed;
4752
// could template name or template path, use for template engine
4853
const { templateSource } = this.apiSchema;
4954
const statement = await this.templateEngine.render(

0 commit comments

Comments
 (0)