@feathersjs/schema 5.0.0-pre.19 → 5.0.0-pre.22
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.
- package/CHANGELOG.md +46 -0
- package/lib/hooks/index.d.ts +2 -0
- package/lib/hooks/index.js +19 -0
- package/lib/hooks/index.js.map +1 -0
- package/lib/hooks/resolve.d.ts +20 -0
- package/lib/hooks/resolve.js +138 -0
- package/lib/hooks/resolve.js.map +1 -0
- package/lib/hooks/validate.d.ts +4 -0
- package/lib/hooks/validate.js +41 -0
- package/lib/hooks/validate.js.map +1 -0
- package/lib/index.d.ts +3 -0
- package/lib/query.d.ts +54 -0
- package/lib/query.js +35 -1
- package/lib/query.js.map +1 -1
- package/lib/resolver.d.ts +3 -0
- package/lib/resolver.js +10 -2
- package/lib/resolver.js.map +1 -1
- package/lib/schema.d.ts +6 -0
- package/lib/schema.js +6 -0
- package/lib/schema.js.map +1 -1
- package/package.json +6 -5
- package/src/hooks/index.ts +2 -0
- package/src/hooks/resolve.ts +185 -0
- package/src/hooks/validate.ts +44 -0
- package/src/index.ts +2 -0
- package/src/query.ts +62 -0
- package/src/resolver.ts +21 -6
- package/src/schema.ts +14 -1
- package/lib/hooks.d.ts +0 -8
- package/lib/hooks.js +0 -119
- package/lib/hooks.js.map +0 -1
- package/src/hooks.ts +0 -143
package/CHANGELOG.md
CHANGED
|
@@ -3,6 +3,52 @@
|
|
|
3
3
|
All notable changes to this project will be documented in this file.
|
|
4
4
|
See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
|
|
5
5
|
|
|
6
|
+
# [5.0.0-pre.22](https://github.com/feathersjs/feathers/compare/v5.0.0-pre.21...v5.0.0-pre.22) (2022-05-24)
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
### Bug Fixes
|
|
10
|
+
|
|
11
|
+
* **schema:** Allows resolveData with different resolvers based on method ([#2644](https://github.com/feathersjs/feathers/issues/2644)) ([be71fa2](https://github.com/feathersjs/feathers/commit/be71fa2fe260e05b7dcc0d5f439e33f2e9ec2434))
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
# [5.0.0-pre.21](https://github.com/feathersjs/feathers/compare/v5.0.0-pre.20...v5.0.0-pre.21) (2022-05-23)
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
### Bug Fixes
|
|
21
|
+
|
|
22
|
+
* **schema:** Add Combine helper to allow merging schema types that use ([#2637](https://github.com/feathersjs/feathers/issues/2637)) ([06d03e9](https://github.com/feathersjs/feathers/commit/06d03e91abb1347576c2351c12322d01c58473e5))
|
|
23
|
+
* **typescript:** Make additional types generic to work with extended types ([#2625](https://github.com/feathersjs/feathers/issues/2625)) ([269fdec](https://github.com/feathersjs/feathers/commit/269fdecc5961092dc8608b3cbe16f433c80bfa96))
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
### Features
|
|
27
|
+
|
|
28
|
+
* **schema:** Add resolveAll hook ([#2643](https://github.com/feathersjs/feathers/issues/2643)) ([85527d7](https://github.com/feathersjs/feathers/commit/85527d71cb78852880696e5d96abdcdf24593934))
|
|
29
|
+
* **schema:** Add resolver for safe external data dispatching ([#2641](https://github.com/feathersjs/feathers/issues/2641)) ([72b980e](https://github.com/feathersjs/feathers/commit/72b980e05631136d30c8f1468dee45ec6a8d2503))
|
|
30
|
+
* **schema:** Add schema resolver converter functionality ([#2640](https://github.com/feathersjs/feathers/issues/2640)) ([26d9e05](https://github.com/feathersjs/feathers/commit/26d9e05327d6e0144466cd57d6fcc11ac7ecb06a))
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
# [5.0.0-pre.20](https://github.com/feathersjs/feathers/compare/v5.0.0-pre.19...v5.0.0-pre.20) (2022-05-04)
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
### Bug Fixes
|
|
40
|
+
|
|
41
|
+
* **dependencies:** Lock monorepo package version numbers ([#2623](https://github.com/feathersjs/feathers/issues/2623)) ([5640c10](https://github.com/feathersjs/feathers/commit/5640c1020cc139994e695d658c08bad3494db507))
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
### Features
|
|
45
|
+
|
|
46
|
+
* **schema:** Add querySyntax helper to create full query schemas ([#2621](https://github.com/feathersjs/feathers/issues/2621)) ([2bbb103](https://github.com/feathersjs/feathers/commit/2bbb103b2f3e30fb0fff935f92ad3276a1a67e41))
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
|
|
6
52
|
# [5.0.0-pre.19](https://github.com/feathersjs/feathers/compare/v5.0.0-pre.18...v5.0.0-pre.19) (2022-05-01)
|
|
7
53
|
|
|
8
54
|
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./resolve"), exports);
|
|
18
|
+
__exportStar(require("./validate"), exports);
|
|
19
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/hooks/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,4CAA0B;AAC1B,6CAA2B"}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
+
import { Resolver } from '../resolver';
|
|
3
|
+
export declare type ResolverSetting<H extends HookContext> = Resolver<any, H> | Resolver<any, H>[];
|
|
4
|
+
export declare type DataResolvers<H extends HookContext> = {
|
|
5
|
+
create: Resolver<any, H>;
|
|
6
|
+
patch: Resolver<any, H>;
|
|
7
|
+
update: Resolver<any, H>;
|
|
8
|
+
};
|
|
9
|
+
export declare type ResolveAllSettings<H extends HookContext> = {
|
|
10
|
+
data?: DataResolvers<H>;
|
|
11
|
+
query?: Resolver<any, H>;
|
|
12
|
+
result?: Resolver<any, H>;
|
|
13
|
+
dispatch?: Resolver<any, H>;
|
|
14
|
+
};
|
|
15
|
+
export declare const DISPATCH: unique symbol;
|
|
16
|
+
export declare const resolveQuery: <T, H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(...resolvers: Resolver<T, H>[]) => (context: H, next?: NextFunction) => Promise<any>;
|
|
17
|
+
export declare const resolveData: <H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(settings: DataResolvers<H> | Resolver<any, H>) => (context: H, next?: NextFunction) => Promise<any>;
|
|
18
|
+
export declare const resolveResult: <T, H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(...resolvers: Resolver<T, H>[]) => (context: H, next?: NextFunction) => Promise<void>;
|
|
19
|
+
export declare const resolveDispatch: <T, H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(...resolvers: Resolver<T, H>[]) => (context: H, next?: NextFunction) => Promise<void>;
|
|
20
|
+
export declare const resolveAll: <H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(map: ResolveAllSettings<H>) => (this: any, context: H, next?: import("@feathersjs/hooks").AsyncMiddleware<H>) => Promise<any>;
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.resolveAll = exports.resolveDispatch = exports.resolveResult = exports.resolveData = exports.resolveQuery = exports.DISPATCH = void 0;
|
|
4
|
+
const hooks_1 = require("@feathersjs/hooks");
|
|
5
|
+
const resolver_1 = require("../resolver");
|
|
6
|
+
const getContext = (context) => {
|
|
7
|
+
return {
|
|
8
|
+
...context,
|
|
9
|
+
params: {
|
|
10
|
+
...context.params,
|
|
11
|
+
query: {}
|
|
12
|
+
}
|
|
13
|
+
};
|
|
14
|
+
};
|
|
15
|
+
const getData = (context) => {
|
|
16
|
+
const isPaginated = context.method === 'find' && context.result.data;
|
|
17
|
+
const data = isPaginated ? context.result.data : context.result;
|
|
18
|
+
return { isPaginated, data };
|
|
19
|
+
};
|
|
20
|
+
const runResolvers = async (resolvers, data, ctx, status) => {
|
|
21
|
+
let current = data;
|
|
22
|
+
for (const resolver of resolvers) {
|
|
23
|
+
current = await resolver.resolve(current, ctx, status);
|
|
24
|
+
}
|
|
25
|
+
return current;
|
|
26
|
+
};
|
|
27
|
+
exports.DISPATCH = Symbol('@feathersjs/schema/dispatch');
|
|
28
|
+
const resolveQuery = (...resolvers) => async (context, next) => {
|
|
29
|
+
var _a;
|
|
30
|
+
const ctx = getContext(context);
|
|
31
|
+
const data = ((_a = context === null || context === void 0 ? void 0 : context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
32
|
+
const query = await runResolvers(resolvers, data, ctx);
|
|
33
|
+
context.params = {
|
|
34
|
+
...context.params,
|
|
35
|
+
query
|
|
36
|
+
};
|
|
37
|
+
if (typeof next === 'function') {
|
|
38
|
+
return next();
|
|
39
|
+
}
|
|
40
|
+
};
|
|
41
|
+
exports.resolveQuery = resolveQuery;
|
|
42
|
+
const resolveData = (settings) => async (context, next) => {
|
|
43
|
+
if (context.method === 'create' || context.method === 'patch' || context.method === 'update') {
|
|
44
|
+
const resolvers = settings instanceof resolver_1.Resolver ? [settings] : [settings[context.method]];
|
|
45
|
+
const ctx = getContext(context);
|
|
46
|
+
const data = context.data;
|
|
47
|
+
const status = {
|
|
48
|
+
originalContext: context
|
|
49
|
+
};
|
|
50
|
+
if (Array.isArray(data)) {
|
|
51
|
+
context.data = await Promise.all(data.map(current => runResolvers(resolvers, current, ctx, status)));
|
|
52
|
+
}
|
|
53
|
+
else {
|
|
54
|
+
context.data = await runResolvers(resolvers, data, ctx, status);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
if (typeof next === 'function') {
|
|
58
|
+
return next();
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
exports.resolveData = resolveData;
|
|
62
|
+
const resolveResult = (...resolvers) => async (context, next) => {
|
|
63
|
+
var _a;
|
|
64
|
+
if (typeof next === 'function') {
|
|
65
|
+
const { $resolve: properties, ...query } = ((_a = context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
66
|
+
const resolve = {
|
|
67
|
+
originalContext: context,
|
|
68
|
+
...context.params.resolve,
|
|
69
|
+
properties
|
|
70
|
+
};
|
|
71
|
+
context.params = {
|
|
72
|
+
...context.params,
|
|
73
|
+
resolve,
|
|
74
|
+
query
|
|
75
|
+
};
|
|
76
|
+
await next();
|
|
77
|
+
}
|
|
78
|
+
const ctx = getContext(context);
|
|
79
|
+
const status = context.params.resolve;
|
|
80
|
+
const { isPaginated, data } = getData(context);
|
|
81
|
+
const result = Array.isArray(data) ?
|
|
82
|
+
await Promise.all(data.map(async (current) => runResolvers(resolvers, current, ctx, status))) :
|
|
83
|
+
await runResolvers(resolvers, data, ctx, status);
|
|
84
|
+
if (isPaginated) {
|
|
85
|
+
context.result.data = result;
|
|
86
|
+
}
|
|
87
|
+
else {
|
|
88
|
+
context.result = result;
|
|
89
|
+
}
|
|
90
|
+
};
|
|
91
|
+
exports.resolveResult = resolveResult;
|
|
92
|
+
const resolveDispatch = (...resolvers) => async (context, next) => {
|
|
93
|
+
if (typeof next === 'function') {
|
|
94
|
+
await next();
|
|
95
|
+
}
|
|
96
|
+
const ctx = getContext(context);
|
|
97
|
+
const status = context.params.resolve;
|
|
98
|
+
const { isPaginated, data } = getData(context);
|
|
99
|
+
const resolveDispatch = async (current) => {
|
|
100
|
+
const resolved = await runResolvers(resolvers, current, ctx, status);
|
|
101
|
+
return Object.keys(resolved).reduce((res, key) => {
|
|
102
|
+
const value = current[key];
|
|
103
|
+
const hasDispatch = typeof value === 'object' && value !== null && value[exports.DISPATCH] !== undefined;
|
|
104
|
+
res[key] = hasDispatch ? value[exports.DISPATCH] : value;
|
|
105
|
+
return res;
|
|
106
|
+
}, {});
|
|
107
|
+
};
|
|
108
|
+
const result = await (Array.isArray(data) ? Promise.all(data.map(resolveDispatch)) : resolveDispatch(data));
|
|
109
|
+
const dispatch = isPaginated ? {
|
|
110
|
+
...context.result,
|
|
111
|
+
data: result
|
|
112
|
+
} : result;
|
|
113
|
+
context.dispatch = dispatch;
|
|
114
|
+
Object.defineProperty(context.result, exports.DISPATCH, {
|
|
115
|
+
value: dispatch,
|
|
116
|
+
enumerable: false,
|
|
117
|
+
configurable: false
|
|
118
|
+
});
|
|
119
|
+
};
|
|
120
|
+
exports.resolveDispatch = resolveDispatch;
|
|
121
|
+
const resolveAll = (map) => {
|
|
122
|
+
const middleware = [];
|
|
123
|
+
if (map.dispatch) {
|
|
124
|
+
middleware.push((0, exports.resolveDispatch)(map.dispatch));
|
|
125
|
+
}
|
|
126
|
+
if (map.result) {
|
|
127
|
+
middleware.push((0, exports.resolveResult)(map.result));
|
|
128
|
+
}
|
|
129
|
+
if (map.query) {
|
|
130
|
+
middleware.push((0, exports.resolveQuery)(map.query));
|
|
131
|
+
}
|
|
132
|
+
if (map.data) {
|
|
133
|
+
middleware.push((0, exports.resolveData)(map.data));
|
|
134
|
+
}
|
|
135
|
+
return (0, hooks_1.compose)(middleware);
|
|
136
|
+
};
|
|
137
|
+
exports.resolveAll = resolveAll;
|
|
138
|
+
//# sourceMappingURL=resolve.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolve.js","sourceRoot":"","sources":["../../src/hooks/resolve.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
+
import { Schema } from '../schema';
|
|
3
|
+
export declare const validateQuery: <H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(schema: Schema<any>) => (context: H, next?: NextFunction) => Promise<any>;
|
|
4
|
+
export declare const validateData: <H extends HookContext<import("@feathersjs/feathers").Application<any, any>, any>>(schema: Schema<any>) => (context: H, next?: NextFunction) => Promise<any>;
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.validateData = exports.validateQuery = void 0;
|
|
4
|
+
const lib_1 = require("../../../errors/lib");
|
|
5
|
+
const validateQuery = (schema) => async (context, next) => {
|
|
6
|
+
var _a;
|
|
7
|
+
const data = ((_a = context === null || context === void 0 ? void 0 : context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
8
|
+
try {
|
|
9
|
+
const query = await schema.validate(data);
|
|
10
|
+
context.params = {
|
|
11
|
+
...context.params,
|
|
12
|
+
query
|
|
13
|
+
};
|
|
14
|
+
if (typeof next === 'function') {
|
|
15
|
+
return next();
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
catch (error) {
|
|
19
|
+
throw (error.ajv ? new lib_1.BadRequest(error.message, error.errors) : error);
|
|
20
|
+
}
|
|
21
|
+
};
|
|
22
|
+
exports.validateQuery = validateQuery;
|
|
23
|
+
const validateData = (schema) => async (context, next) => {
|
|
24
|
+
const data = context.data;
|
|
25
|
+
try {
|
|
26
|
+
if (Array.isArray(data)) {
|
|
27
|
+
context.data = await Promise.all(data.map(current => schema.validate(current)));
|
|
28
|
+
}
|
|
29
|
+
else {
|
|
30
|
+
context.data = await schema.validate(data);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
throw (error.ajv ? new lib_1.BadRequest(error.message, error.errors) : error);
|
|
35
|
+
}
|
|
36
|
+
if (typeof next === 'function') {
|
|
37
|
+
return next();
|
|
38
|
+
}
|
|
39
|
+
};
|
|
40
|
+
exports.validateData = validateData;
|
|
41
|
+
//# sourceMappingURL=validate.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validate.js","sourceRoot":"","sources":["../../src/hooks/validate.ts"],"names":[],"mappings":";;;AACA,6CAAiD;AAG1C,MAAM,aAAa,GAAG,CAAyB,MAAmB,EAAE,EAAE,CAC3E,KAAK,EAAE,OAAU,EAAE,IAAmB,EAAE,EAAE;;IACxC,MAAM,IAAI,GAAG,CAAA,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,MAAM,0CAAE,KAAK,KAAI,EAAE,CAAC;IAE1C,IAAI;QACF,MAAM,KAAK,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QAE1C,OAAO,CAAC,MAAM,GAAG;YACf,GAAG,OAAO,CAAC,MAAM;YACjB,KAAK;SACN,CAAA;QAED,IAAI,OAAO,IAAI,KAAK,UAAU,EAAE;YAC9B,OAAO,IAAI,EAAE,CAAC;SACf;KACF;IAAC,OAAO,KAAU,EAAE;QACnB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,gBAAU,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;KACzE;AACH,CAAC,CAAC;AAlBS,QAAA,aAAa,iBAkBtB;AAEK,MAAM,YAAY,GAAG,CAAyB,MAAmB,EAAE,EAAE,CAC5E,KAAK,EAAE,OAAU,EAAE,IAAmB,EAAE,EAAE;IACxC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;IAE1B,IAAI;QACF,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACvB,OAAO,CAAC,IAAI,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAClD,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,CACzB,CAAC,CAAC;SACJ;aAAM;YACL,OAAO,CAAC,IAAI,GAAG,MAAM,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;SAC5C;KACF;IAAC,OAAO,KAAU,EAAE;QACnB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,gBAAU,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;KACzE;IAED,IAAI,OAAO,IAAI,KAAK,UAAU,EAAE;QAC9B,OAAO,IAAI,EAAE,CAAC;KACf;AACH,CAAC,CAAC;AAnBW,QAAA,YAAY,gBAmBvB"}
|
package/lib/index.d.ts
CHANGED
|
@@ -6,6 +6,9 @@ export * from './query';
|
|
|
6
6
|
export declare type Infer<S extends {
|
|
7
7
|
_type: any;
|
|
8
8
|
}> = S['_type'];
|
|
9
|
+
export declare type Combine<S extends {
|
|
10
|
+
_type: any;
|
|
11
|
+
}, U> = Pick<Infer<S>, Exclude<keyof Infer<S>, keyof U>> & U;
|
|
9
12
|
declare module '@feathersjs/feathers/lib/declarations' {
|
|
10
13
|
interface Params {
|
|
11
14
|
resolve?: ResolverStatus<any, HookContext>;
|
package/lib/query.d.ts
CHANGED
|
@@ -1,3 +1,28 @@
|
|
|
1
|
+
import { JSONSchema } from 'json-schema-to-ts';
|
|
2
|
+
export declare type PropertyQuery<D extends JSONSchema> = {
|
|
3
|
+
anyOf: [
|
|
4
|
+
D,
|
|
5
|
+
{
|
|
6
|
+
type: 'object';
|
|
7
|
+
additionalProperties: false;
|
|
8
|
+
properties: {
|
|
9
|
+
$gt: D;
|
|
10
|
+
$gte: D;
|
|
11
|
+
$lt: D;
|
|
12
|
+
$lte: D;
|
|
13
|
+
$ne: D;
|
|
14
|
+
$in: {
|
|
15
|
+
type: 'array';
|
|
16
|
+
items: D;
|
|
17
|
+
};
|
|
18
|
+
$nin: {
|
|
19
|
+
type: 'array';
|
|
20
|
+
items: D;
|
|
21
|
+
};
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
];
|
|
25
|
+
};
|
|
1
26
|
export declare const queryProperty: <T extends import("json-schema-to-ts").JSONSchema7>(definition: T) => {
|
|
2
27
|
readonly anyOf: readonly [T, {
|
|
3
28
|
readonly type: "object";
|
|
@@ -19,3 +44,32 @@ export declare const queryProperty: <T extends import("json-schema-to-ts").JSONS
|
|
|
19
44
|
};
|
|
20
45
|
}];
|
|
21
46
|
};
|
|
47
|
+
export declare const queryProperties: <T extends {
|
|
48
|
+
[key: string]: import("json-schema-to-ts").JSONSchema7;
|
|
49
|
+
}>(definition: T) => { [K in keyof T]: PropertyQuery<T[K]>; };
|
|
50
|
+
export declare const querySyntax: <T extends {
|
|
51
|
+
[key: string]: import("json-schema-to-ts").JSONSchema7;
|
|
52
|
+
}>(definition: T) => {
|
|
53
|
+
readonly $limit: {
|
|
54
|
+
readonly type: "number";
|
|
55
|
+
readonly minimum: 0;
|
|
56
|
+
};
|
|
57
|
+
readonly $skip: {
|
|
58
|
+
readonly type: "number";
|
|
59
|
+
readonly minimum: 0;
|
|
60
|
+
};
|
|
61
|
+
readonly $sort: {
|
|
62
|
+
readonly type: "object";
|
|
63
|
+
readonly properties: { [K in keyof T]: {
|
|
64
|
+
readonly type: 'number';
|
|
65
|
+
readonly enum: [1, -1];
|
|
66
|
+
}; };
|
|
67
|
+
};
|
|
68
|
+
readonly $select: {
|
|
69
|
+
readonly type: "array";
|
|
70
|
+
readonly items: {
|
|
71
|
+
readonly type: "string";
|
|
72
|
+
readonly enum: (keyof T)[];
|
|
73
|
+
};
|
|
74
|
+
};
|
|
75
|
+
} & { [K_1 in keyof T]: PropertyQuery<T[K_1]>; };
|
package/lib/query.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.queryProperty = void 0;
|
|
3
|
+
exports.querySyntax = exports.queryProperties = exports.queryProperty = void 0;
|
|
4
4
|
const queryProperty = (definition) => ({
|
|
5
5
|
anyOf: [
|
|
6
6
|
definition,
|
|
@@ -26,4 +26,38 @@ const queryProperty = (definition) => ({
|
|
|
26
26
|
]
|
|
27
27
|
});
|
|
28
28
|
exports.queryProperty = queryProperty;
|
|
29
|
+
const queryProperties = (definition) => Object.keys(definition).reduce((res, key) => {
|
|
30
|
+
res[key] = (0, exports.queryProperty)(definition[key]);
|
|
31
|
+
return res;
|
|
32
|
+
}, {});
|
|
33
|
+
exports.queryProperties = queryProperties;
|
|
34
|
+
const querySyntax = (definition) => ({
|
|
35
|
+
$limit: {
|
|
36
|
+
type: 'number',
|
|
37
|
+
minimum: 0
|
|
38
|
+
},
|
|
39
|
+
$skip: {
|
|
40
|
+
type: 'number',
|
|
41
|
+
minimum: 0
|
|
42
|
+
},
|
|
43
|
+
$sort: {
|
|
44
|
+
type: 'object',
|
|
45
|
+
properties: Object.keys(definition).reduce((res, key) => {
|
|
46
|
+
res[key] = {
|
|
47
|
+
type: 'number',
|
|
48
|
+
enum: [1, -1]
|
|
49
|
+
};
|
|
50
|
+
return res;
|
|
51
|
+
}, {})
|
|
52
|
+
},
|
|
53
|
+
$select: {
|
|
54
|
+
type: 'array',
|
|
55
|
+
items: {
|
|
56
|
+
type: 'string',
|
|
57
|
+
enum: Object.keys(definition)
|
|
58
|
+
}
|
|
59
|
+
},
|
|
60
|
+
...(0, exports.queryProperties)(definition)
|
|
61
|
+
});
|
|
62
|
+
exports.querySyntax = querySyntax;
|
|
29
63
|
//# sourceMappingURL=query.js.map
|
package/lib/query.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"query.js","sourceRoot":"","sources":["../src/query.ts"],"names":[],"mappings":";;;
|
|
1
|
+
{"version":3,"file":"query.js","sourceRoot":"","sources":["../src/query.ts"],"names":[],"mappings":";;;AA2BO,MAAM,aAAa,GAAG,CAAwB,UAAa,EAAE,EAAE,CAAC,CAAC;IACtE,KAAK,EAAE;QACL,UAAU;QACV;YACE,IAAI,EAAE,QAAQ;YACd,oBAAoB,EAAE,KAAK;YAC3B,UAAU,EAAE;gBACV,GAAG,EAAE,UAAU;gBACf,IAAI,EAAE,UAAU;gBAChB,GAAG,EAAE,UAAU;gBACf,IAAI,EAAE,UAAU;gBAChB,GAAG,EAAE,UAAU;gBACf,GAAG,EAAE;oBACH,IAAI,EAAE,OAAO;oBACb,KAAK,EAAE,UAAU;iBAClB;gBACD,IAAI,EAAE;oBACJ,IAAI,EAAE,OAAO;oBACb,KAAK,EAAE,UAAU;iBAClB;aACF;SACF;KACF;CACQ,CAAA,CAAC;AAvBC,QAAA,aAAa,iBAuBd;AAEL,MAAM,eAAe,GAAG,CAA2C,UAAa,EAAE,EAAE,CACzF,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;IACzC,GAAW,CAAC,GAAG,CAAC,GAAG,IAAA,qBAAa,EAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAA;IAElD,OAAO,GAAG,CAAA;AACZ,CAAC,EAAE,EAA6C,CAAC,CAAA;AALtC,QAAA,eAAe,mBAKuB;AAE5C,MAAM,WAAW,GAAG,CAA2C,UAAa,EAAE,EAAE,CAAC,CAAC;IACvF,MAAM,EAAE;QACN,IAAI,EAAE,QAAQ;QACd,OAAO,EAAE,CAAC;KACX;IACD,KAAK,EAAE;QACL,IAAI,EAAE,QAAQ;QACd,OAAO,EAAE,CAAC;KACX;IACD,KAAK,EAAE;QACL,IAAI,EAAE,QAAQ;QACd,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;YACrD,GAAW,CAAC,GAAG,CAAC,GAAG;gBAClB,IAAI,EAAE,QAAQ;gBACd,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;aACd,CAAA;YAED,OAAO,GAAG,CAAA;QACZ,CAAC,EAAE,EAA6E,CAAC;KAClF;IACD,OAAO,EAAE;QACP,IAAI,EAAE,OAAO;QACb,KAAK,EAAE;YACL,IAAI,EAAE,QAAQ;YACd,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAuB;SACpD;KACF;IACD,GAAG,IAAA,uBAAe,EAAC,UAAU,CAAC;CACrB,CAAA,CAAA;AA5BE,QAAA,WAAW,eA4Bb"}
|
package/lib/resolver.d.ts
CHANGED
|
@@ -3,10 +3,12 @@ export declare type PropertyResolver<T, V, C> = (value: V | undefined, obj: T, c
|
|
|
3
3
|
export declare type PropertyResolverMap<T, C> = {
|
|
4
4
|
[key in keyof T]?: PropertyResolver<T, T[key], C>;
|
|
5
5
|
};
|
|
6
|
+
export declare type ResolverConverter<T, C> = (obj: any, context: C, status: ResolverStatus<T, C>) => Promise<T | undefined>;
|
|
6
7
|
export interface ResolverConfig<T, C> {
|
|
7
8
|
schema?: Schema<T>;
|
|
8
9
|
validate?: 'before' | 'after' | false;
|
|
9
10
|
properties: PropertyResolverMap<T, C>;
|
|
11
|
+
converter?: ResolverConverter<T, C>;
|
|
10
12
|
}
|
|
11
13
|
export interface ResolverStatus<T, C> {
|
|
12
14
|
path: string[];
|
|
@@ -19,6 +21,7 @@ export declare class Resolver<T, C> {
|
|
|
19
21
|
readonly _type: T;
|
|
20
22
|
constructor(options: ResolverConfig<T, C>);
|
|
21
23
|
resolveProperty<D, K extends keyof T>(name: K, data: D, context: C, status?: Partial<ResolverStatus<T, C>>): Promise<T[K]>;
|
|
24
|
+
convert<D>(data: D, context: C, status?: Partial<ResolverStatus<T, C>>): Promise<T | D>;
|
|
22
25
|
resolve<D>(_data: D, context: C, status?: Partial<ResolverStatus<T, C>>): Promise<T>;
|
|
23
26
|
}
|
|
24
27
|
export declare function resolve<T, C>(options: ResolverConfig<T, C>): Resolver<T, C>;
|
package/lib/resolver.js
CHANGED
|
@@ -21,9 +21,17 @@ class Resolver {
|
|
|
21
21
|
};
|
|
22
22
|
return resolver(value, data, context, resolverStatus);
|
|
23
23
|
}
|
|
24
|
+
async convert(data, context, status) {
|
|
25
|
+
if (this.options.converter) {
|
|
26
|
+
const { path = [], stack = [] } = status || {};
|
|
27
|
+
return this.options.converter(data, context, { ...status, path, stack });
|
|
28
|
+
}
|
|
29
|
+
return data;
|
|
30
|
+
}
|
|
24
31
|
async resolve(_data, context, status) {
|
|
25
32
|
const { properties: resolvers, schema, validate } = this.options;
|
|
26
|
-
const
|
|
33
|
+
const payload = await this.convert(_data, context, status);
|
|
34
|
+
const data = schema && validate === 'before' ? await schema.validate(payload) : payload;
|
|
27
35
|
const propertyList = (Array.isArray(status === null || status === void 0 ? void 0 : status.properties)
|
|
28
36
|
? status === null || status === void 0 ? void 0 : status.properties
|
|
29
37
|
// By default get all data and resolver keys but remove duplicates
|
|
@@ -56,7 +64,7 @@ class Resolver {
|
|
|
56
64
|
}));
|
|
57
65
|
if (hasErrors) {
|
|
58
66
|
const propertyName = (status === null || status === void 0 ? void 0 : status.properties) ? ` ${status.properties.join('.')}` : '';
|
|
59
|
-
throw new errors_1.BadRequest('Error resolving data' + propertyName, errors);
|
|
67
|
+
throw new errors_1.BadRequest('Error resolving data' + (propertyName ? ` ${propertyName}` : ''), errors);
|
|
60
68
|
}
|
|
61
69
|
return schema && validate === 'after'
|
|
62
70
|
? await schema.validate(result)
|
package/lib/resolver.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../src/resolver.ts"],"names":[],"mappings":";;;AAAA,+CAAgD;
|
|
1
|
+
{"version":3,"file":"resolver.js","sourceRoot":"","sources":["../src/resolver.ts"],"names":[],"mappings":"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"}
|
package/lib/schema.d.ts
CHANGED
|
@@ -5,6 +5,10 @@ export { Ajv };
|
|
|
5
5
|
export declare type JSONSchemaDefinition = JSONSchema & {
|
|
6
6
|
$id: string;
|
|
7
7
|
$async?: boolean;
|
|
8
|
+
properties?: {
|
|
9
|
+
[key: string]: JSONSchema;
|
|
10
|
+
};
|
|
11
|
+
required?: readonly string[];
|
|
8
12
|
};
|
|
9
13
|
export interface Schema<T> {
|
|
10
14
|
validate<X = T>(...args: Parameters<ValidateFunction<X>>): Promise<X>;
|
|
@@ -15,6 +19,8 @@ export declare class SchemaWrapper<S extends JSONSchemaDefinition> implements Sc
|
|
|
15
19
|
validator: AsyncValidateFunction;
|
|
16
20
|
readonly _type: FromSchema<S>;
|
|
17
21
|
constructor(definition: S, ajv?: Ajv);
|
|
22
|
+
get properties(): S["properties"];
|
|
23
|
+
get required(): S["required"];
|
|
18
24
|
validate<T = FromSchema<S>>(...args: Parameters<ValidateFunction<T>>): Promise<T>;
|
|
19
25
|
toJSON(): S;
|
|
20
26
|
}
|
package/lib/schema.js
CHANGED
|
@@ -19,6 +19,12 @@ class SchemaWrapper {
|
|
|
19
19
|
...this.definition
|
|
20
20
|
});
|
|
21
21
|
}
|
|
22
|
+
get properties() {
|
|
23
|
+
return this.definition.properties;
|
|
24
|
+
}
|
|
25
|
+
get required() {
|
|
26
|
+
return this.definition.required;
|
|
27
|
+
}
|
|
22
28
|
async validate(...args) {
|
|
23
29
|
try {
|
|
24
30
|
const validated = await this.validator(...args);
|
package/lib/schema.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"schema.js","sourceRoot":"","sources":["../src/schema.ts"],"names":[],"mappings":";;;;;;AAAA,8CAAmE;AAQ1D,cARF,aAAG,CAQE;AANZ,+CAAgD;AAEnC,QAAA,WAAW,GAAG,IAAI,aAAG,CAAC;IACjC,WAAW,EAAE,IAAI;CAClB,CAAC,CAAC;
|
|
1
|
+
{"version":3,"file":"schema.js","sourceRoot":"","sources":["../src/schema.ts"],"names":[],"mappings":";;;;;;AAAA,8CAAmE;AAQ1D,cARF,aAAG,CAQE;AANZ,+CAAgD;AAEnC,QAAA,WAAW,GAAG,IAAI,aAAG,CAAC;IACjC,WAAW,EAAE,IAAI;CAClB,CAAC,CAAC;AAeH,MAAa,aAAa;IAKxB,YAAoB,UAAa,EAAE,MAAW,mBAAW;QAArC,eAAU,GAAV,UAAU,CAAG;QAC/B,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC;QACf,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC;YAChC,MAAM,EAAE,IAAI;YACZ,GAAI,IAAI,CAAC,UAAkB;SAC5B,CAA0B,CAAC;IAC9B,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,UAAU,CAAC,UAA6B,CAAC;IACvD,CAAC;IAED,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,UAAU,CAAC,QAAyB,CAAC;IACnD,CAAC;IAED,KAAK,CAAC,QAAQ,CAAsB,GAAG,IAAqC;QAC1E,IAAI;YACF,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,GAAG,IAAI,CAAM,CAAC;YAErD,OAAO,SAAS,CAAC;SAClB;QAAC,OAAO,KAAU,EAAE;YACnB,MAAM,IAAI,mBAAU,CAAC,KAAK,CAAC,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC;SACnD;IACH,CAAC;IAED,MAAM;QACJ,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;CACF;AAlCD,sCAkCC;AAED,SAAgB,MAAM,CAAmC,UAAa,EAAE,MAAW,mBAAW;IAC5F,OAAO,IAAI,aAAa,CAAC,UAAU,EAAE,GAAG,CAAC,CAAC;AAC5C,CAAC;AAFD,wBAEC"}
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@feathersjs/schema",
|
|
3
3
|
"description": "A common data schema definition format",
|
|
4
|
-
"version": "5.0.0-pre.
|
|
4
|
+
"version": "5.0.0-pre.22",
|
|
5
5
|
"homepage": "https://feathersjs.com",
|
|
6
6
|
"main": "lib/",
|
|
7
7
|
"types": "lib/",
|
|
@@ -53,15 +53,16 @@
|
|
|
53
53
|
"access": "public"
|
|
54
54
|
},
|
|
55
55
|
"dependencies": {
|
|
56
|
-
"@feathersjs/errors": "^5.0.0-pre.
|
|
57
|
-
"@feathersjs/feathers": "^5.0.0-pre.
|
|
56
|
+
"@feathersjs/errors": "^5.0.0-pre.22",
|
|
57
|
+
"@feathersjs/feathers": "^5.0.0-pre.22",
|
|
58
|
+
"@feathersjs/hooks": "^0.7.4",
|
|
58
59
|
"@types/json-schema": "^7.0.11",
|
|
59
60
|
"ajv": "^8.11.0",
|
|
60
61
|
"json-schema": "^0.4.0",
|
|
61
62
|
"json-schema-to-ts": "^2.3.0"
|
|
62
63
|
},
|
|
63
64
|
"devDependencies": {
|
|
64
|
-
"@feathersjs/memory": "^5.0.0-pre.
|
|
65
|
+
"@feathersjs/memory": "^5.0.0-pre.22",
|
|
65
66
|
"@types/mocha": "^9.1.1",
|
|
66
67
|
"@types/node": "^17.0.31",
|
|
67
68
|
"ajv-formats": "^2.1.1",
|
|
@@ -69,5 +70,5 @@
|
|
|
69
70
|
"shx": "^0.3.4",
|
|
70
71
|
"typescript": "^4.6.4"
|
|
71
72
|
},
|
|
72
|
-
"gitHead": "
|
|
73
|
+
"gitHead": "e452e02063e6d8943a9cae2315ab585bc4f82fb6"
|
|
73
74
|
}
|
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
+
import { compose } from '@feathersjs/hooks';
|
|
3
|
+
import { Resolver, ResolverStatus } from '../resolver';
|
|
4
|
+
|
|
5
|
+
const getContext = <H extends HookContext> (context: H) => {
|
|
6
|
+
return {
|
|
7
|
+
...context,
|
|
8
|
+
params: {
|
|
9
|
+
...context.params,
|
|
10
|
+
query: {}
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
const getData = <H extends HookContext> (context: H) => {
|
|
16
|
+
const isPaginated = context.method === 'find' && context.result.data;
|
|
17
|
+
const data = isPaginated ? context.result.data : context.result;
|
|
18
|
+
|
|
19
|
+
return { isPaginated, data };
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const runResolvers = async <T, H extends HookContext> (
|
|
23
|
+
resolvers: Resolver<T, H>[],
|
|
24
|
+
data: any,
|
|
25
|
+
ctx: H,
|
|
26
|
+
status?: Partial<ResolverStatus<T, H>>
|
|
27
|
+
) => {
|
|
28
|
+
let current: any = data;
|
|
29
|
+
|
|
30
|
+
for (const resolver of resolvers) {
|
|
31
|
+
current = await resolver.resolve(current, ctx, status);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
return current as T;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export type ResolverSetting<H extends HookContext> = Resolver<any, H>|Resolver<any, H>[];
|
|
38
|
+
|
|
39
|
+
export type DataResolvers<H extends HookContext> = {
|
|
40
|
+
create: Resolver<any, H>
|
|
41
|
+
patch: Resolver<any, H>
|
|
42
|
+
update: Resolver<any, H>
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export type ResolveAllSettings<H extends HookContext> = {
|
|
46
|
+
data?: DataResolvers<H>
|
|
47
|
+
query?: Resolver<any, H>
|
|
48
|
+
result?: Resolver<any, H>
|
|
49
|
+
dispatch?: Resolver<any, H>
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export const DISPATCH = Symbol('@feathersjs/schema/dispatch');
|
|
53
|
+
|
|
54
|
+
export const resolveQuery = <T, H extends HookContext> (...resolvers: Resolver<T, H>[]) =>
|
|
55
|
+
async (context: H, next?: NextFunction) => {
|
|
56
|
+
const ctx = getContext(context);
|
|
57
|
+
const data = context?.params?.query || {};
|
|
58
|
+
const query = await runResolvers(resolvers, data, ctx);
|
|
59
|
+
|
|
60
|
+
context.params = {
|
|
61
|
+
...context.params,
|
|
62
|
+
query
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
if (typeof next === 'function') {
|
|
66
|
+
return next();
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
export const resolveData = <H extends HookContext> (settings: DataResolvers<H>|Resolver<any, H>) =>
|
|
71
|
+
async (context: H, next?: NextFunction) => {
|
|
72
|
+
if (context.method === 'create' || context.method === 'patch' || context.method === 'update') {
|
|
73
|
+
const resolvers = settings instanceof Resolver ? [ settings ] : [ settings[context.method] ];
|
|
74
|
+
const ctx = getContext(context);
|
|
75
|
+
const data = context.data;
|
|
76
|
+
|
|
77
|
+
const status = {
|
|
78
|
+
originalContext: context
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
if (Array.isArray(data)) {
|
|
82
|
+
context.data = await Promise.all(data.map(current =>
|
|
83
|
+
runResolvers(resolvers, current, ctx, status)
|
|
84
|
+
));
|
|
85
|
+
} else {
|
|
86
|
+
context.data = await runResolvers(resolvers, data, ctx, status);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (typeof next === 'function') {
|
|
91
|
+
return next();
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
|
|
95
|
+
export const resolveResult = <T, H extends HookContext> (...resolvers: Resolver<T, H>[]) =>
|
|
96
|
+
async (context: H, next?: NextFunction) => {
|
|
97
|
+
if (typeof next === 'function') {
|
|
98
|
+
const { $resolve: properties, ...query } = context.params?.query || {};
|
|
99
|
+
const resolve = {
|
|
100
|
+
originalContext: context,
|
|
101
|
+
...context.params.resolve,
|
|
102
|
+
properties
|
|
103
|
+
};
|
|
104
|
+
|
|
105
|
+
context.params = {
|
|
106
|
+
...context.params,
|
|
107
|
+
resolve,
|
|
108
|
+
query
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
await next();
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
const ctx = getContext(context);
|
|
115
|
+
const status = context.params.resolve;
|
|
116
|
+
const { isPaginated, data } = getData(context);
|
|
117
|
+
|
|
118
|
+
const result = Array.isArray(data) ?
|
|
119
|
+
await Promise.all(data.map(async current => runResolvers(resolvers, current, ctx, status))) :
|
|
120
|
+
await runResolvers(resolvers, data, ctx, status);
|
|
121
|
+
|
|
122
|
+
if (isPaginated) {
|
|
123
|
+
context.result.data = result;
|
|
124
|
+
} else {
|
|
125
|
+
context.result = result;
|
|
126
|
+
}
|
|
127
|
+
};
|
|
128
|
+
|
|
129
|
+
export const resolveDispatch = <T, H extends HookContext> (...resolvers: Resolver<T, H>[]) =>
|
|
130
|
+
async (context: H, next?: NextFunction) => {
|
|
131
|
+
if (typeof next === 'function') {
|
|
132
|
+
await next();
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
const ctx = getContext(context);
|
|
136
|
+
const status = context.params.resolve;
|
|
137
|
+
const { isPaginated, data } = getData(context);
|
|
138
|
+
const resolveDispatch = async (current: any) => {
|
|
139
|
+
const resolved = await runResolvers(resolvers, current, ctx, status)
|
|
140
|
+
|
|
141
|
+
return Object.keys(resolved).reduce((res, key) => {
|
|
142
|
+
const value = current[key];
|
|
143
|
+
const hasDispatch = typeof value === 'object' && value !== null && value[DISPATCH] !== undefined;
|
|
144
|
+
|
|
145
|
+
res[key] = hasDispatch ? value[DISPATCH] : value;
|
|
146
|
+
|
|
147
|
+
return res
|
|
148
|
+
}, {} as any)
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const result = await (Array.isArray(data) ? Promise.all(data.map(resolveDispatch)) : resolveDispatch(data));
|
|
152
|
+
const dispatch = isPaginated ? {
|
|
153
|
+
...context.result,
|
|
154
|
+
data: result
|
|
155
|
+
} : result;
|
|
156
|
+
|
|
157
|
+
context.dispatch = dispatch;
|
|
158
|
+
Object.defineProperty(context.result, DISPATCH, {
|
|
159
|
+
value: dispatch,
|
|
160
|
+
enumerable: false,
|
|
161
|
+
configurable: false
|
|
162
|
+
});
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
export const resolveAll = <H extends HookContext> (map: ResolveAllSettings<H>) => {
|
|
166
|
+
const middleware = [];
|
|
167
|
+
|
|
168
|
+
if (map.dispatch) {
|
|
169
|
+
middleware.push(resolveDispatch(map.dispatch));
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
if (map.result) {
|
|
173
|
+
middleware.push(resolveResult(map.result));
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
if (map.query) {
|
|
177
|
+
middleware.push(resolveQuery(map.query));
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
if (map.data) {
|
|
181
|
+
middleware.push(resolveData(map.data));
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
return compose(middleware);
|
|
185
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
+
import { BadRequest } from '../../../errors/lib';
|
|
3
|
+
import { Schema } from '../schema';
|
|
4
|
+
|
|
5
|
+
export const validateQuery = <H extends HookContext> (schema: Schema<any>) =>
|
|
6
|
+
async (context: H, next?: NextFunction) => {
|
|
7
|
+
const data = context?.params?.query || {};
|
|
8
|
+
|
|
9
|
+
try {
|
|
10
|
+
const query = await schema.validate(data);
|
|
11
|
+
|
|
12
|
+
context.params = {
|
|
13
|
+
...context.params,
|
|
14
|
+
query
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
if (typeof next === 'function') {
|
|
18
|
+
return next();
|
|
19
|
+
}
|
|
20
|
+
} catch (error: any) {
|
|
21
|
+
throw (error.ajv ? new BadRequest(error.message, error.errors) : error);
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
export const validateData = <H extends HookContext> (schema: Schema<any>) =>
|
|
26
|
+
async (context: H, next?: NextFunction) => {
|
|
27
|
+
const data = context.data;
|
|
28
|
+
|
|
29
|
+
try {
|
|
30
|
+
if (Array.isArray(data)) {
|
|
31
|
+
context.data = await Promise.all(data.map(current =>
|
|
32
|
+
schema.validate(current)
|
|
33
|
+
));
|
|
34
|
+
} else {
|
|
35
|
+
context.data = await schema.validate(data);
|
|
36
|
+
}
|
|
37
|
+
} catch (error: any) {
|
|
38
|
+
throw (error.ajv ? new BadRequest(error.message, error.errors) : error);
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
if (typeof next === 'function') {
|
|
42
|
+
return next();
|
|
43
|
+
}
|
|
44
|
+
};
|
package/src/index.ts
CHANGED
|
@@ -7,6 +7,8 @@ export * from './query';
|
|
|
7
7
|
|
|
8
8
|
export type Infer<S extends { _type: any }> = S['_type'];
|
|
9
9
|
|
|
10
|
+
export type Combine<S extends { _type: any }, U> = Pick<Infer<S>, Exclude<keyof Infer<S>, keyof U>> & U;
|
|
11
|
+
|
|
10
12
|
declare module '@feathersjs/feathers/lib/declarations' {
|
|
11
13
|
interface Params {
|
|
12
14
|
resolve?: ResolverStatus<any, HookContext>;
|
package/src/query.ts
CHANGED
|
@@ -1,5 +1,30 @@
|
|
|
1
1
|
import { JSONSchema } from 'json-schema-to-ts';
|
|
2
2
|
|
|
3
|
+
export type PropertyQuery<D extends JSONSchema> = {
|
|
4
|
+
anyOf: [
|
|
5
|
+
D,
|
|
6
|
+
{
|
|
7
|
+
type: 'object',
|
|
8
|
+
additionalProperties: false,
|
|
9
|
+
properties: {
|
|
10
|
+
$gt: D,
|
|
11
|
+
$gte: D,
|
|
12
|
+
$lt: D,
|
|
13
|
+
$lte: D,
|
|
14
|
+
$ne: D,
|
|
15
|
+
$in: {
|
|
16
|
+
type: 'array',
|
|
17
|
+
items: D
|
|
18
|
+
},
|
|
19
|
+
$nin: {
|
|
20
|
+
type: 'array',
|
|
21
|
+
items: D
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
]
|
|
26
|
+
}
|
|
27
|
+
|
|
3
28
|
export const queryProperty = <T extends JSONSchema> (definition: T) => ({
|
|
4
29
|
anyOf: [
|
|
5
30
|
definition,
|
|
@@ -24,3 +49,40 @@ export const queryProperty = <T extends JSONSchema> (definition: T) => ({
|
|
|
24
49
|
}
|
|
25
50
|
]
|
|
26
51
|
} as const);
|
|
52
|
+
|
|
53
|
+
export const queryProperties = <T extends { [key: string]: JSONSchema }> (definition: T) =>
|
|
54
|
+
Object.keys(definition).reduce((res, key) => {
|
|
55
|
+
(res as any)[key] = queryProperty(definition[key])
|
|
56
|
+
|
|
57
|
+
return res
|
|
58
|
+
}, {} as { [K in keyof T]: PropertyQuery<T[K]> })
|
|
59
|
+
|
|
60
|
+
export const querySyntax = <T extends { [key: string]: JSONSchema }> (definition: T) => ({
|
|
61
|
+
$limit: {
|
|
62
|
+
type: 'number',
|
|
63
|
+
minimum: 0
|
|
64
|
+
},
|
|
65
|
+
$skip: {
|
|
66
|
+
type: 'number',
|
|
67
|
+
minimum: 0
|
|
68
|
+
},
|
|
69
|
+
$sort: {
|
|
70
|
+
type: 'object',
|
|
71
|
+
properties: Object.keys(definition).reduce((res, key) => {
|
|
72
|
+
(res as any)[key] = {
|
|
73
|
+
type: 'number',
|
|
74
|
+
enum: [1, -1]
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
return res
|
|
78
|
+
}, {} as { [K in keyof T]: { readonly type: 'number', readonly enum: [1, -1] } })
|
|
79
|
+
},
|
|
80
|
+
$select: {
|
|
81
|
+
type: 'array',
|
|
82
|
+
items: {
|
|
83
|
+
type: 'string',
|
|
84
|
+
enum: Object.keys(definition) as any as (keyof T)[]
|
|
85
|
+
}
|
|
86
|
+
},
|
|
87
|
+
...queryProperties(definition)
|
|
88
|
+
} as const)
|
package/src/resolver.ts
CHANGED
|
@@ -12,17 +12,21 @@ export type PropertyResolverMap<T, C> = {
|
|
|
12
12
|
[key in keyof T]?: PropertyResolver<T, T[key], C>
|
|
13
13
|
}
|
|
14
14
|
|
|
15
|
+
export type ResolverConverter<T, C> = (obj: any, context: C, status: ResolverStatus<T, C>)
|
|
16
|
+
=> Promise<T|undefined>
|
|
17
|
+
|
|
15
18
|
export interface ResolverConfig<T, C> {
|
|
16
19
|
schema?: Schema<T>,
|
|
17
20
|
validate?: 'before'|'after'|false,
|
|
18
21
|
properties: PropertyResolverMap<T, C>
|
|
22
|
+
converter?: ResolverConverter<T, C>
|
|
19
23
|
}
|
|
20
24
|
|
|
21
25
|
export interface ResolverStatus<T, C> {
|
|
22
|
-
path: string[]
|
|
23
|
-
originalContext?: C
|
|
24
|
-
properties?: string[]
|
|
25
|
-
stack: PropertyResolver<T, any, C>[]
|
|
26
|
+
path: string[]
|
|
27
|
+
originalContext?: C
|
|
28
|
+
properties?: string[]
|
|
29
|
+
stack: PropertyResolver<T, any, C>[]
|
|
26
30
|
}
|
|
27
31
|
|
|
28
32
|
export class Resolver<T, C> {
|
|
@@ -55,9 +59,20 @@ export class Resolver<T, C> {
|
|
|
55
59
|
return resolver(value, data as any, context, resolverStatus);
|
|
56
60
|
}
|
|
57
61
|
|
|
62
|
+
async convert <D> (data: D, context: C, status?: Partial<ResolverStatus<T, C>>) {
|
|
63
|
+
if (this.options.converter) {
|
|
64
|
+
const { path = [], stack = [] } = status || {}
|
|
65
|
+
|
|
66
|
+
return this.options.converter(data, context, { ...status, path, stack })
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return data
|
|
70
|
+
}
|
|
71
|
+
|
|
58
72
|
async resolve<D> (_data: D, context: C, status?: Partial<ResolverStatus<T, C>>): Promise<T> {
|
|
59
73
|
const { properties: resolvers, schema, validate } = this.options;
|
|
60
|
-
const
|
|
74
|
+
const payload = await this.convert(_data, context, status);
|
|
75
|
+
const data = schema && validate === 'before' ? await schema.validate(payload) : payload;
|
|
61
76
|
const propertyList = (Array.isArray(status?.properties)
|
|
62
77
|
? status?.properties
|
|
63
78
|
// By default get all data and resolver keys but remove duplicates
|
|
@@ -96,7 +111,7 @@ export class Resolver<T, C> {
|
|
|
96
111
|
if (hasErrors) {
|
|
97
112
|
const propertyName = status?.properties ? ` ${status.properties.join('.')}` : '';
|
|
98
113
|
|
|
99
|
-
throw new BadRequest('Error resolving data' + propertyName, errors);
|
|
114
|
+
throw new BadRequest('Error resolving data' + (propertyName ? ` ${propertyName}` : ''), errors);
|
|
100
115
|
}
|
|
101
116
|
|
|
102
117
|
return schema && validate === 'after'
|
package/src/schema.ts
CHANGED
|
@@ -8,7 +8,12 @@ export const DEFAULT_AJV = new Ajv({
|
|
|
8
8
|
|
|
9
9
|
export { Ajv };
|
|
10
10
|
|
|
11
|
-
export type JSONSchemaDefinition = JSONSchema & {
|
|
11
|
+
export type JSONSchemaDefinition = JSONSchema & {
|
|
12
|
+
$id: string,
|
|
13
|
+
$async?: boolean,
|
|
14
|
+
properties?: { [key: string]: JSONSchema }
|
|
15
|
+
required?: readonly string[]
|
|
16
|
+
};
|
|
12
17
|
|
|
13
18
|
export interface Schema<T> {
|
|
14
19
|
validate <X = T> (...args: Parameters<ValidateFunction<X>>): Promise<X>;
|
|
@@ -27,6 +32,14 @@ export class SchemaWrapper<S extends JSONSchemaDefinition> implements Schema<Fro
|
|
|
27
32
|
}) as AsyncValidateFunction;
|
|
28
33
|
}
|
|
29
34
|
|
|
35
|
+
get properties () {
|
|
36
|
+
return this.definition.properties as S['properties'];
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
get required () {
|
|
40
|
+
return this.definition.required as S['required'];
|
|
41
|
+
}
|
|
42
|
+
|
|
30
43
|
async validate <T = FromSchema<S>> (...args: Parameters<ValidateFunction<T>>) {
|
|
31
44
|
try {
|
|
32
45
|
const validated = await this.validator(...args) as T;
|
package/lib/hooks.d.ts
DELETED
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
-
import { Resolver } from './resolver';
|
|
3
|
-
import { Schema } from './schema';
|
|
4
|
-
export declare const resolveQuery: <T>(...resolvers: Resolver<T, HookContext<import("@feathersjs/feathers").Application<any, any>, any>>[]) => (context: HookContext, next?: NextFunction) => Promise<any>;
|
|
5
|
-
export declare const resolveData: <T>(...resolvers: Resolver<T, HookContext<import("@feathersjs/feathers").Application<any, any>, any>>[]) => (context: HookContext, next?: NextFunction) => Promise<any>;
|
|
6
|
-
export declare const resolveResult: <T>(...resolvers: Resolver<T, HookContext<import("@feathersjs/feathers").Application<any, any>, any>>[]) => (context: HookContext, next?: NextFunction) => Promise<void>;
|
|
7
|
-
export declare const validateQuery: (schema: Schema<any>) => (context: HookContext, next?: NextFunction) => Promise<any>;
|
|
8
|
-
export declare const validateData: (schema: Schema<any>) => (context: HookContext, next?: NextFunction) => Promise<any>;
|
package/lib/hooks.js
DELETED
|
@@ -1,119 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.validateData = exports.validateQuery = exports.resolveResult = exports.resolveData = exports.resolveQuery = void 0;
|
|
4
|
-
const lib_1 = require("../../errors/lib");
|
|
5
|
-
const getContext = (context) => {
|
|
6
|
-
return {
|
|
7
|
-
...context,
|
|
8
|
-
params: {
|
|
9
|
-
...context.params,
|
|
10
|
-
query: {}
|
|
11
|
-
}
|
|
12
|
-
};
|
|
13
|
-
};
|
|
14
|
-
const runResolvers = async (resolvers, data, ctx, status) => {
|
|
15
|
-
let current = data;
|
|
16
|
-
for (const resolver of resolvers) {
|
|
17
|
-
current = await resolver.resolve(current, ctx, status);
|
|
18
|
-
}
|
|
19
|
-
return current;
|
|
20
|
-
};
|
|
21
|
-
const resolveQuery = (...resolvers) => async (context, next) => {
|
|
22
|
-
var _a;
|
|
23
|
-
const ctx = getContext(context);
|
|
24
|
-
const data = ((_a = context === null || context === void 0 ? void 0 : context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
25
|
-
const query = await runResolvers(resolvers, data, ctx);
|
|
26
|
-
context.params = {
|
|
27
|
-
...context.params,
|
|
28
|
-
query
|
|
29
|
-
};
|
|
30
|
-
if (typeof next === 'function') {
|
|
31
|
-
return next();
|
|
32
|
-
}
|
|
33
|
-
};
|
|
34
|
-
exports.resolveQuery = resolveQuery;
|
|
35
|
-
const resolveData = (...resolvers) => async (context, next) => {
|
|
36
|
-
const ctx = getContext(context);
|
|
37
|
-
const data = context.data;
|
|
38
|
-
const status = {
|
|
39
|
-
originalContext: context
|
|
40
|
-
};
|
|
41
|
-
if (Array.isArray(data)) {
|
|
42
|
-
context.data = await Promise.all(data.map(current => runResolvers(resolvers, current, ctx, status)));
|
|
43
|
-
}
|
|
44
|
-
else {
|
|
45
|
-
context.data = await runResolvers(resolvers, data, ctx, status);
|
|
46
|
-
}
|
|
47
|
-
if (typeof next === 'function') {
|
|
48
|
-
return next();
|
|
49
|
-
}
|
|
50
|
-
};
|
|
51
|
-
exports.resolveData = resolveData;
|
|
52
|
-
const resolveResult = (...resolvers) => async (context, next) => {
|
|
53
|
-
var _a;
|
|
54
|
-
if (typeof next === 'function') {
|
|
55
|
-
const { $resolve: properties, ...query } = ((_a = context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
56
|
-
const resolve = {
|
|
57
|
-
originalContext: context,
|
|
58
|
-
...context.params.resolve,
|
|
59
|
-
properties
|
|
60
|
-
};
|
|
61
|
-
context.params = {
|
|
62
|
-
...context.params,
|
|
63
|
-
resolve,
|
|
64
|
-
query
|
|
65
|
-
};
|
|
66
|
-
await next();
|
|
67
|
-
}
|
|
68
|
-
const ctx = getContext(context);
|
|
69
|
-
const status = context.params.resolve;
|
|
70
|
-
const isPaginated = context.method === 'find' && context.result.data;
|
|
71
|
-
const data = isPaginated ? context.result.data : context.result;
|
|
72
|
-
const result = Array.isArray(data) ?
|
|
73
|
-
await Promise.all(data.map(async (current) => runResolvers(resolvers, current, ctx, status))) :
|
|
74
|
-
await runResolvers(resolvers, data, ctx, status);
|
|
75
|
-
if (isPaginated) {
|
|
76
|
-
context.result.data = result;
|
|
77
|
-
}
|
|
78
|
-
else {
|
|
79
|
-
context.result = result;
|
|
80
|
-
}
|
|
81
|
-
};
|
|
82
|
-
exports.resolveResult = resolveResult;
|
|
83
|
-
const validateQuery = (schema) => async (context, next) => {
|
|
84
|
-
var _a;
|
|
85
|
-
const data = ((_a = context === null || context === void 0 ? void 0 : context.params) === null || _a === void 0 ? void 0 : _a.query) || {};
|
|
86
|
-
try {
|
|
87
|
-
const query = await schema.validate(data);
|
|
88
|
-
context.params = {
|
|
89
|
-
...context.params,
|
|
90
|
-
query
|
|
91
|
-
};
|
|
92
|
-
if (typeof next === 'function') {
|
|
93
|
-
return next();
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
catch (error) {
|
|
97
|
-
throw (error.ajv ? new lib_1.BadRequest(error.message, error.errors) : error);
|
|
98
|
-
}
|
|
99
|
-
};
|
|
100
|
-
exports.validateQuery = validateQuery;
|
|
101
|
-
const validateData = (schema) => async (context, next) => {
|
|
102
|
-
const data = context.data;
|
|
103
|
-
try {
|
|
104
|
-
if (Array.isArray(data)) {
|
|
105
|
-
context.data = await Promise.all(data.map(current => schema.validate(current)));
|
|
106
|
-
}
|
|
107
|
-
else {
|
|
108
|
-
context.data = await schema.validate(data);
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
catch (error) {
|
|
112
|
-
throw (error.ajv ? new lib_1.BadRequest(error.message, error.errors) : error);
|
|
113
|
-
}
|
|
114
|
-
if (typeof next === 'function') {
|
|
115
|
-
return next();
|
|
116
|
-
}
|
|
117
|
-
};
|
|
118
|
-
exports.validateData = validateData;
|
|
119
|
-
//# sourceMappingURL=hooks.js.map
|
package/lib/hooks.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"hooks.js","sourceRoot":"","sources":["../src/hooks.ts"],"names":[],"mappings":"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"}
|
package/src/hooks.ts
DELETED
|
@@ -1,143 +0,0 @@
|
|
|
1
|
-
import { HookContext, NextFunction } from '@feathersjs/feathers';
|
|
2
|
-
import { BadRequest } from '../../errors/lib';
|
|
3
|
-
import { Resolver, ResolverStatus } from './resolver';
|
|
4
|
-
import { Schema } from './schema';
|
|
5
|
-
|
|
6
|
-
const getContext = (context: HookContext) => {
|
|
7
|
-
return {
|
|
8
|
-
...context,
|
|
9
|
-
params: {
|
|
10
|
-
...context.params,
|
|
11
|
-
query: {}
|
|
12
|
-
}
|
|
13
|
-
}
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
const runResolvers = async <T> (
|
|
17
|
-
resolvers: Resolver<T, HookContext>[],
|
|
18
|
-
data: any,
|
|
19
|
-
ctx: HookContext,
|
|
20
|
-
status?: Partial<ResolverStatus<T, HookContext>>
|
|
21
|
-
) => {
|
|
22
|
-
let current: any = data;
|
|
23
|
-
|
|
24
|
-
for (const resolver of resolvers) {
|
|
25
|
-
current = await resolver.resolve(current, ctx, status);
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
return current as T;
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
export const resolveQuery = <T> (...resolvers: Resolver<T, HookContext>[]) =>
|
|
32
|
-
async (context: HookContext, next?: NextFunction) => {
|
|
33
|
-
const ctx = getContext(context);
|
|
34
|
-
const data = context?.params?.query || {};
|
|
35
|
-
const query = await runResolvers(resolvers, data, ctx);
|
|
36
|
-
|
|
37
|
-
context.params = {
|
|
38
|
-
...context.params,
|
|
39
|
-
query
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
if (typeof next === 'function') {
|
|
43
|
-
return next();
|
|
44
|
-
}
|
|
45
|
-
};
|
|
46
|
-
|
|
47
|
-
export const resolveData = <T> (...resolvers: Resolver<T, HookContext>[]) =>
|
|
48
|
-
async (context: HookContext, next?: NextFunction) => {
|
|
49
|
-
const ctx = getContext(context);
|
|
50
|
-
const data = context.data;
|
|
51
|
-
const status = {
|
|
52
|
-
originalContext: context
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
if (Array.isArray(data)) {
|
|
56
|
-
context.data = await Promise.all(data.map(current =>
|
|
57
|
-
runResolvers(resolvers, current, ctx, status)
|
|
58
|
-
));
|
|
59
|
-
} else {
|
|
60
|
-
context.data = await runResolvers(resolvers, data, ctx, status);
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
if (typeof next === 'function') {
|
|
64
|
-
return next();
|
|
65
|
-
}
|
|
66
|
-
};
|
|
67
|
-
|
|
68
|
-
export const resolveResult = <T> (...resolvers: Resolver<T, HookContext>[]) =>
|
|
69
|
-
async (context: HookContext, next?: NextFunction) => {
|
|
70
|
-
if (typeof next === 'function') {
|
|
71
|
-
const { $resolve: properties, ...query } = context.params?.query || {};
|
|
72
|
-
const resolve = {
|
|
73
|
-
originalContext: context,
|
|
74
|
-
...context.params.resolve,
|
|
75
|
-
properties
|
|
76
|
-
};
|
|
77
|
-
|
|
78
|
-
context.params = {
|
|
79
|
-
...context.params,
|
|
80
|
-
resolve,
|
|
81
|
-
query
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
await next();
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
const ctx = getContext(context);
|
|
88
|
-
const status = context.params.resolve;
|
|
89
|
-
|
|
90
|
-
const isPaginated = context.method === 'find' && context.result.data;
|
|
91
|
-
const data = isPaginated ? context.result.data : context.result;
|
|
92
|
-
|
|
93
|
-
const result = Array.isArray(data) ?
|
|
94
|
-
await Promise.all(data.map(async current => runResolvers(resolvers, current, ctx, status))) :
|
|
95
|
-
await runResolvers(resolvers, data, ctx, status);
|
|
96
|
-
|
|
97
|
-
if (isPaginated) {
|
|
98
|
-
context.result.data = result;
|
|
99
|
-
} else {
|
|
100
|
-
context.result = result;
|
|
101
|
-
}
|
|
102
|
-
};
|
|
103
|
-
|
|
104
|
-
export const validateQuery = (schema: Schema<any>) =>
|
|
105
|
-
async (context: HookContext, next?: NextFunction) => {
|
|
106
|
-
const data = context?.params?.query || {};
|
|
107
|
-
|
|
108
|
-
try {
|
|
109
|
-
const query = await schema.validate(data);
|
|
110
|
-
|
|
111
|
-
context.params = {
|
|
112
|
-
...context.params,
|
|
113
|
-
query
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
if (typeof next === 'function') {
|
|
117
|
-
return next();
|
|
118
|
-
}
|
|
119
|
-
} catch (error: any) {
|
|
120
|
-
throw (error.ajv ? new BadRequest(error.message, error.errors) : error);
|
|
121
|
-
}
|
|
122
|
-
};
|
|
123
|
-
|
|
124
|
-
export const validateData = (schema: Schema<any>) =>
|
|
125
|
-
async (context: HookContext, next?: NextFunction) => {
|
|
126
|
-
const data = context.data;
|
|
127
|
-
|
|
128
|
-
try {
|
|
129
|
-
if (Array.isArray(data)) {
|
|
130
|
-
context.data = await Promise.all(data.map(current =>
|
|
131
|
-
schema.validate(current)
|
|
132
|
-
));
|
|
133
|
-
} else {
|
|
134
|
-
context.data = await schema.validate(data);
|
|
135
|
-
}
|
|
136
|
-
} catch (error: any) {
|
|
137
|
-
throw (error.ajv ? new BadRequest(error.message, error.errors) : error);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
if (typeof next === 'function') {
|
|
141
|
-
return next();
|
|
142
|
-
}
|
|
143
|
-
};
|