@adaptivestone/framework 5.0.0-beta.11 → 5.0.0-beta.13
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 +10 -0
- package/LICENCE +1 -1
- package/cliCommand.ts +10 -0
- package/dist/Cli.d.ts +7 -0
- package/dist/Cli.js +19 -0
- package/dist/Cli.js.map +1 -0
- package/dist/cluster.d.ts +1 -0
- package/dist/cluster.js +18 -0
- package/dist/cluster.js.map +1 -0
- package/dist/commands/CreateUser.d.ts +32 -0
- package/dist/commands/CreateUser.js +81 -0
- package/dist/commands/CreateUser.js.map +1 -0
- package/dist/commands/Documentation.d.ts +5 -0
- package/dist/commands/Documentation.js +15 -0
- package/dist/commands/Documentation.js.map +1 -0
- package/dist/commands/DropIndex.d.ts +16 -0
- package/dist/commands/DropIndex.js +30 -0
- package/dist/commands/DropIndex.js.map +1 -0
- package/dist/commands/GenerateRandomBytes.d.ts +7 -0
- package/dist/commands/GenerateRandomBytes.js +18 -0
- package/dist/commands/GenerateRandomBytes.js.map +1 -0
- package/dist/commands/GetOpenApiJson.d.ts +17 -0
- package/dist/commands/GetOpenApiJson.js +272 -0
- package/dist/commands/GetOpenApiJson.js.map +1 -0
- package/dist/commands/SyncIndexes.d.ts +6 -0
- package/dist/commands/SyncIndexes.js +32 -0
- package/dist/commands/SyncIndexes.js.map +1 -0
- package/dist/commands/migration/Create.d.ts +18 -0
- package/dist/commands/migration/Create.js +57 -0
- package/dist/commands/migration/Create.js.map +1 -0
- package/dist/commands/migration/Migrate.d.ts +6 -0
- package/dist/commands/migration/Migrate.js +46 -0
- package/dist/commands/migration/Migrate.js.map +1 -0
- package/dist/config/auth.d.ts +6 -0
- package/dist/config/auth.js +7 -0
- package/dist/config/auth.js.map +1 -0
- package/dist/config/http.d.ts +8 -0
- package/dist/config/http.js +10 -0
- package/dist/config/http.js.map +1 -0
- package/dist/config/i18n.d.ts +10 -0
- package/dist/config/i18n.js +13 -0
- package/dist/config/i18n.js.map +1 -0
- package/dist/config/ipDetector.d.ts +5 -0
- package/dist/config/ipDetector.js +15 -0
- package/dist/config/ipDetector.js.map +1 -0
- package/dist/config/log.d.ts +22 -0
- package/dist/config/log.js +23 -0
- package/dist/config/log.js.map +1 -0
- package/dist/config/mongo.d.ts +4 -0
- package/dist/config/mongo.js +4 -0
- package/dist/config/mongo.js.map +1 -0
- package/dist/config/rateLimiter.d.ts +15 -0
- package/dist/config/rateLimiter.js +17 -0
- package/dist/config/rateLimiter.js.map +1 -0
- package/dist/config/redis.d.ts +5 -0
- package/dist/config/redis.js +5 -0
- package/dist/config/redis.js.map +1 -0
- package/dist/config/validate.d.ts +4 -0
- package/dist/config/validate.js +4 -0
- package/dist/config/validate.js.map +1 -0
- package/dist/controllers/Auth.d.ts +74 -0
- package/dist/controllers/Auth.js +181 -0
- package/dist/controllers/Auth.js.map +1 -0
- package/dist/controllers/Home.d.ts +15 -0
- package/dist/controllers/Home.js +24 -0
- package/dist/controllers/Home.js.map +1 -0
- package/dist/controllers/index.d.ts +16 -0
- package/dist/controllers/index.js +59 -0
- package/dist/controllers/index.js.map +1 -0
- package/dist/folderConfig.d.ts +34 -0
- package/dist/folderConfig.js +15 -0
- package/dist/folderConfig.js.map +1 -0
- package/dist/helpers/files.d.ts +16 -0
- package/dist/helpers/files.js +76 -0
- package/dist/helpers/files.js.map +1 -0
- package/dist/helpers/logger.d.ts +4 -0
- package/dist/helpers/logger.js +28 -0
- package/dist/helpers/logger.js.map +1 -0
- package/dist/helpers/redis/clearNamespace.d.ts +3 -0
- package/dist/helpers/redis/clearNamespace.js +15 -0
- package/dist/helpers/redis/clearNamespace.js.map +1 -0
- package/dist/helpers/yup.d.ts +13 -0
- package/{helpers → dist/helpers}/yup.js +8 -9
- package/dist/helpers/yup.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +7 -0
- package/dist/index.js.map +1 -0
- package/dist/models/Lock.d.ts +62 -0
- package/dist/models/Lock.js +94 -0
- package/dist/models/Lock.js.map +1 -0
- package/dist/models/Migration.d.ts +13 -0
- package/dist/models/Migration.js +14 -0
- package/dist/models/Migration.js.map +1 -0
- package/dist/models/Sequence.d.ts +23 -0
- package/dist/models/Sequence.js +16 -0
- package/dist/models/Sequence.js.map +1 -0
- package/dist/models/User.d.ts +178 -0
- package/dist/models/User.js +244 -0
- package/dist/models/User.js.map +1 -0
- package/dist/modules/AbstractCommand.d.ts +30 -0
- package/dist/modules/AbstractCommand.js +44 -0
- package/dist/modules/AbstractCommand.js.map +1 -0
- package/dist/modules/AbstractConnector.d.ts +5 -0
- package/dist/modules/AbstractConnector.js +8 -0
- package/dist/modules/AbstractConnector.js.map +1 -0
- package/dist/modules/AbstractController.d.ts +90 -0
- package/dist/modules/AbstractController.js +324 -0
- package/dist/modules/AbstractController.js.map +1 -0
- package/dist/modules/AbstractModel.d.ts +29 -0
- package/dist/modules/AbstractModel.js +73 -0
- package/dist/modules/AbstractModel.js.map +1 -0
- package/dist/modules/Base.d.ts +29 -0
- package/dist/modules/Base.js +54 -0
- package/dist/modules/Base.js.map +1 -0
- package/dist/modules/BaseCli.d.ts +25 -0
- package/dist/modules/BaseCli.js +138 -0
- package/dist/modules/BaseCli.js.map +1 -0
- package/dist/server.d.ts +110 -0
- package/dist/server.js +372 -0
- package/dist/server.js.map +1 -0
- package/dist/services/cache/Cache.d.ts +31 -0
- package/dist/services/cache/Cache.js +115 -0
- package/dist/services/cache/Cache.js.map +1 -0
- package/dist/services/documentation/DocumentationGenerator.d.ts +11 -0
- package/dist/services/documentation/DocumentationGenerator.js +134 -0
- package/dist/services/documentation/DocumentationGenerator.js.map +1 -0
- package/dist/services/http/HttpServer.d.ts +35 -0
- package/dist/services/http/HttpServer.js +71 -0
- package/dist/services/http/HttpServer.js.map +1 -0
- package/dist/services/http/middleware/AbstractMiddleware.d.ts +25 -0
- package/dist/services/http/middleware/AbstractMiddleware.js +43 -0
- package/dist/services/http/middleware/AbstractMiddleware.js.map +1 -0
- package/dist/services/http/middleware/Auth.d.ts +9 -0
- package/dist/services/http/middleware/Auth.js +18 -0
- package/dist/services/http/middleware/Auth.js.map +1 -0
- package/dist/services/http/middleware/Cors.d.ts +10 -0
- package/dist/services/http/middleware/Cors.js +36 -0
- package/dist/services/http/middleware/Cors.js.map +1 -0
- package/dist/services/http/middleware/GetUserByToken.d.ts +19 -0
- package/dist/services/http/middleware/GetUserByToken.js +40 -0
- package/dist/services/http/middleware/GetUserByToken.js.map +1 -0
- package/dist/services/http/middleware/I18n.d.ts +26 -0
- package/dist/services/http/middleware/I18n.js +97 -0
- package/dist/services/http/middleware/I18n.js.map +1 -0
- package/dist/services/http/middleware/IpDetector.d.ts +12 -0
- package/dist/services/http/middleware/IpDetector.js +51 -0
- package/dist/services/http/middleware/IpDetector.js.map +1 -0
- package/dist/services/http/middleware/Pagination.d.ts +27 -0
- package/dist/services/http/middleware/Pagination.js +47 -0
- package/dist/services/http/middleware/Pagination.js.map +1 -0
- package/dist/services/http/middleware/PrepareAppInfo.d.ts +8 -0
- package/dist/services/http/middleware/PrepareAppInfo.js +17 -0
- package/dist/services/http/middleware/PrepareAppInfo.js.map +1 -0
- package/dist/services/http/middleware/RateLimiter.d.ts +18 -0
- package/dist/services/http/middleware/RateLimiter.js +106 -0
- package/dist/services/http/middleware/RateLimiter.js.map +1 -0
- package/dist/services/http/middleware/RequestLogger.d.ts +8 -0
- package/dist/services/http/middleware/RequestLogger.js +18 -0
- package/dist/services/http/middleware/RequestLogger.js.map +1 -0
- package/dist/services/http/middleware/RequestParser.d.ts +8 -0
- package/dist/services/http/middleware/RequestParser.js +35 -0
- package/dist/services/http/middleware/RequestParser.js.map +1 -0
- package/dist/services/http/middleware/Role.d.ts +9 -0
- package/dist/services/http/middleware/Role.js +24 -0
- package/dist/services/http/middleware/Role.js.map +1 -0
- package/dist/services/validate/ValidateService.d.ts +31 -0
- package/dist/services/validate/ValidateService.js +96 -0
- package/dist/services/validate/ValidateService.js.map +1 -0
- package/dist/services/validate/drivers/AbstractValidator.d.ts +14 -0
- package/dist/services/validate/drivers/AbstractValidator.js +32 -0
- package/dist/services/validate/drivers/AbstractValidator.js.map +1 -0
- package/dist/services/validate/drivers/CustomValidator.d.ts +14 -0
- package/dist/services/validate/drivers/CustomValidator.js +48 -0
- package/dist/services/validate/drivers/CustomValidator.js.map +1 -0
- package/dist/services/validate/drivers/YupValidator.d.ts +13 -0
- package/dist/services/validate/drivers/YupValidator.js +86 -0
- package/dist/services/validate/drivers/YupValidator.js.map +1 -0
- package/dist/tests/globalSetupVitest.d.ts +2 -0
- package/dist/tests/globalSetupVitest.js +29 -0
- package/dist/tests/globalSetupVitest.js.map +1 -0
- package/dist/tests/setupVitest.d.ts +1 -0
- package/dist/tests/setupVitest.js +99 -0
- package/dist/tests/setupVitest.js.map +1 -0
- package/package.json +21 -11
- package/Cli.js +0 -21
- package/cluster.js +0 -25
- package/commands/.gitkeep +0 -0
- package/commands/CreateUser.js +0 -106
- package/commands/Documentation.js +0 -17
- package/commands/DropIndex.js +0 -38
- package/commands/GenerateRandomBytes.js +0 -21
- package/commands/GetOpenApiJson.js +0 -336
- package/commands/SyncIndexes.js +0 -40
- package/commands/migration/Create.js +0 -72
- package/commands/migration/Migrate.js +0 -56
- package/config/auth.js +0 -9
- package/config/http.js +0 -9
- package/config/i18n.js +0 -12
- package/config/ipDetector.js +0 -14
- package/config/log.js +0 -22
- package/config/mongo.js +0 -3
- package/config/rateLimiter.js +0 -16
- package/config/redis.js +0 -4
- package/config/validate.js +0 -3
- package/controllers/Auth.js +0 -201
- package/controllers/Home.js +0 -28
- package/controllers/index.js +0 -66
- package/eslint.config.js +0 -68
- package/folderConfig.js +0 -13
- package/helpers/files.js +0 -76
- package/helpers/logger.js +0 -16
- package/helpers/redis/clearNamespace.js +0 -14
- package/index.js +0 -8
- package/jsconfig.json +0 -9
- package/locales/en/translation.json +0 -27
- package/locales/ru/translation.json +0 -27
- package/migrations/.gitkeep +0 -0
- package/models/Lock.js +0 -107
- package/models/Migration.js +0 -15
- package/models/Sequence.js +0 -22
- package/models/User.js +0 -280
- package/modules/AbstractCommand.d.ts +0 -50
- package/modules/AbstractCommand.js +0 -52
- package/modules/AbstractConnector.js +0 -9
- package/modules/AbstractController.d.ts +0 -55
- package/modules/AbstractController.js +0 -400
- package/modules/AbstractModel.d.ts +0 -48
- package/modules/AbstractModel.js +0 -92
- package/modules/Base.d.ts +0 -37
- package/modules/Base.js +0 -64
- package/modules/BaseCli.js +0 -162
- package/server.d.ts +0 -100
- package/server.js +0 -439
- package/services/cache/Cache.d.ts +0 -35
- package/services/cache/Cache.js +0 -124
- package/services/documentation/DocumentationGenerator.js +0 -169
- package/services/http/HttpServer.js +0 -89
- package/services/http/middleware/AbstractMiddleware.js +0 -51
- package/services/http/middleware/Auth.js +0 -20
- package/services/http/middleware/Cors.js +0 -46
- package/services/http/middleware/GetUserByToken.js +0 -47
- package/services/http/middleware/I18n.js +0 -117
- package/services/http/middleware/IpDetector.js +0 -59
- package/services/http/middleware/Pagination.js +0 -57
- package/services/http/middleware/PrepareAppInfo.js +0 -18
- package/services/http/middleware/RateLimiter.js +0 -134
- package/services/http/middleware/RequestLogger.js +0 -22
- package/services/http/middleware/RequestParser.js +0 -40
- package/services/http/middleware/Role.js +0 -29
- package/services/messaging/email/resources/.gitkeep +0 -1
- package/services/messaging/email/templates/.gitkeep +0 -0
- package/services/messaging/email/templates/recovery/html.pug +0 -8
- package/services/messaging/email/templates/recovery/subject.pug +0 -2
- package/services/messaging/email/templates/recovery/text.pug +0 -3
- package/services/messaging/email/templates/verification/html.pug +0 -10
- package/services/messaging/email/templates/verification/subject.pug +0 -1
- package/services/messaging/email/templates/verification/text.pug +0 -1
- package/services/validate/ValidateService.js +0 -157
- package/services/validate/drivers/AbstractValidator.js +0 -37
- package/services/validate/drivers/CustomValidator.js +0 -51
- package/services/validate/drivers/YupValidator.js +0 -103
- package/tests/globalSetupVitest.js +0 -35
- package/tests/setup.js +0 -120
- package/tests/setupVitest.js +0 -111
- package/types/Expand.d.ts +0 -11
- package/types/ICommandArguments.d.ts +0 -41
- package/types/TFoldersConfig.d.ts +0 -20
- package/views/404.pug +0 -3
- package/views/home.pug +0 -3
- package/views/layouts/base.pug +0 -39
- package/vitest.config.js +0 -17
|
@@ -1,35 +0,0 @@
|
|
|
1
|
-
import Base from '../../modules/Base.d.ts';
|
|
2
|
-
import Server from '../../server.js';
|
|
3
|
-
|
|
4
|
-
declare class Cache extends Base {
|
|
5
|
-
app: Server['app'];
|
|
6
|
-
|
|
7
|
-
constructor(app: Server['app']);
|
|
8
|
-
|
|
9
|
-
/**
|
|
10
|
-
* As framework support namespaces all key for cache go through this function
|
|
11
|
-
* Function return new key with added namespace
|
|
12
|
-
* @param key key to add namespace
|
|
13
|
-
*/
|
|
14
|
-
getKeyWithNameSpace(key: string): string;
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Get value from cache. Set and get if not eists
|
|
18
|
-
* @param key key to check
|
|
19
|
-
* @param onNotFound callback that will be executed if value not found on cahce
|
|
20
|
-
* @param storeTime how long we should store value on cache
|
|
21
|
-
*/
|
|
22
|
-
getSetValue(
|
|
23
|
-
key: string,
|
|
24
|
-
onNotFound: () => Promise<any>,
|
|
25
|
-
storeTime: number,
|
|
26
|
-
): Promise<any>;
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* Remove key from cache
|
|
30
|
-
* @param key key to remove
|
|
31
|
-
*/
|
|
32
|
-
removeKey(key: string): Promise<number>;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
export default Cache;
|
package/services/cache/Cache.js
DELETED
|
@@ -1,124 +0,0 @@
|
|
|
1
|
-
import Base from '../../modules/Base.js';
|
|
2
|
-
|
|
3
|
-
class Cache extends Base {
|
|
4
|
-
constructor(app) {
|
|
5
|
-
super(app);
|
|
6
|
-
this.whenReady = this.#init();
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
async #init() {
|
|
10
|
-
// todo for now only redis. refactor for drives support in future
|
|
11
|
-
// at least memory and redis drivers should be presented
|
|
12
|
-
// memory drives should works on master process level
|
|
13
|
-
// we should support multiple cashe same time
|
|
14
|
-
const { createClient } = await import('@redis/client');
|
|
15
|
-
const conf = this.app.getConfig('redis');
|
|
16
|
-
this.redisClient = createClient({
|
|
17
|
-
url: conf.url,
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
this.redisNamespace = conf.namespace;
|
|
21
|
-
|
|
22
|
-
this.redisClient.on('error', (error, b, c) => {
|
|
23
|
-
this.logger.error(error, b, c);
|
|
24
|
-
});
|
|
25
|
-
this.redisClient.on('connect', () => {
|
|
26
|
-
this.logger.info('Redis connection success');
|
|
27
|
-
});
|
|
28
|
-
this.app.events.on('shutdown', () => {
|
|
29
|
-
this.redisClient.quit();
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
this.promiseMapping = new Map();
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
getKeyWithNameSpace(key) {
|
|
36
|
-
return `${this.redisNamespace}-${key}`;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
async getSetValue(keyValue, onNotFound, storeTime = 60 * 5) {
|
|
40
|
-
await this.whenReady;
|
|
41
|
-
if (!this.redisClient.isOpen) {
|
|
42
|
-
await this.redisClient.connect();
|
|
43
|
-
}
|
|
44
|
-
const key = this.getKeyWithNameSpace(keyValue);
|
|
45
|
-
// 5 mins default
|
|
46
|
-
// eslint-disable-next-line no-unused-vars
|
|
47
|
-
let resolve = (value) => {};
|
|
48
|
-
// eslint-disable-next-line no-unused-vars
|
|
49
|
-
let reject = (value) => {};
|
|
50
|
-
if (this.promiseMapping.has(key)) {
|
|
51
|
-
return this.promiseMapping.get(key);
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
this.promiseMapping.set(
|
|
55
|
-
key,
|
|
56
|
-
new Promise((res, rej) => {
|
|
57
|
-
resolve = res;
|
|
58
|
-
reject = rej;
|
|
59
|
-
}),
|
|
60
|
-
);
|
|
61
|
-
|
|
62
|
-
let result = await this.redisClient.get(key);
|
|
63
|
-
if (!result) {
|
|
64
|
-
this.logger.verbose(`getSetValueFromCache not found for key ${key}`);
|
|
65
|
-
try {
|
|
66
|
-
result = await onNotFound();
|
|
67
|
-
} catch (e) {
|
|
68
|
-
this.logger.error(`Cache onNotFound for key '${key}' error: ${e}`);
|
|
69
|
-
this.promiseMapping.delete(key);
|
|
70
|
-
reject(e);
|
|
71
|
-
return Promise.reject(e);
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
this.redisClient.set(
|
|
75
|
-
key,
|
|
76
|
-
JSON.stringify(result, (_jsonkey, value) =>
|
|
77
|
-
typeof value === 'bigint' ? `${value}n` : value,
|
|
78
|
-
),
|
|
79
|
-
{
|
|
80
|
-
EX: storeTime,
|
|
81
|
-
},
|
|
82
|
-
);
|
|
83
|
-
} else {
|
|
84
|
-
this.logger.verbose(
|
|
85
|
-
`getSetValueFromCache FROM CACHE key ${key}, value ${result.substring(
|
|
86
|
-
0,
|
|
87
|
-
100,
|
|
88
|
-
)}`,
|
|
89
|
-
);
|
|
90
|
-
try {
|
|
91
|
-
result = JSON.parse(result, (_jsonkey, value) => {
|
|
92
|
-
if (typeof value === 'string' && /^\d+n$/.test(value)) {
|
|
93
|
-
return BigInt(value.slice(0, value.length - 1));
|
|
94
|
-
}
|
|
95
|
-
return value;
|
|
96
|
-
});
|
|
97
|
-
} catch {
|
|
98
|
-
this.logger.warn(
|
|
99
|
-
'Not able to parse json from redis cache. That can be a normal in case you store string here',
|
|
100
|
-
);
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
resolve(result);
|
|
105
|
-
this.promiseMapping.delete(key);
|
|
106
|
-
return result;
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
async removeKey(keyValue) {
|
|
110
|
-
await this.whenReady;
|
|
111
|
-
|
|
112
|
-
if (!this.redisClient.isOpen) {
|
|
113
|
-
await this.redisClient.connect();
|
|
114
|
-
}
|
|
115
|
-
const key = this.getKeyWithNameSpace(keyValue);
|
|
116
|
-
return this.redisClient.del(key);
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
static get loggerGroup() {
|
|
120
|
-
return 'Cache_';
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
export default Cache;
|
|
@@ -1,169 +0,0 @@
|
|
|
1
|
-
import Base from '../../modules/Base.js';
|
|
2
|
-
import ValidateService from '../validate/ValidateService.js';
|
|
3
|
-
|
|
4
|
-
class DocumentationGenerator extends Base {
|
|
5
|
-
// eslint-disable-next-line class-methods-use-this
|
|
6
|
-
processingFields(fieldsByRoute) {
|
|
7
|
-
const fields = [];
|
|
8
|
-
if (!fieldsByRoute) {
|
|
9
|
-
return fields;
|
|
10
|
-
}
|
|
11
|
-
const entries = Object.entries(fieldsByRoute);
|
|
12
|
-
entries.forEach(([key, value]) => {
|
|
13
|
-
const field = {};
|
|
14
|
-
field.name = key;
|
|
15
|
-
field.type = value.type;
|
|
16
|
-
if (value.exclusiveTests) {
|
|
17
|
-
field.required = value.exclusiveTests.required;
|
|
18
|
-
}
|
|
19
|
-
if (value?.innerType) {
|
|
20
|
-
field.innerType = value?.innerType?.type;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
if (value.fields) {
|
|
24
|
-
field.fields = [];
|
|
25
|
-
// eslint-disable-next-line no-shadow
|
|
26
|
-
const entries = Object.entries(value.fields);
|
|
27
|
-
// eslint-disable-next-line no-shadow
|
|
28
|
-
entries.forEach(([key, value]) => {
|
|
29
|
-
field.fields.push({
|
|
30
|
-
name: key,
|
|
31
|
-
type: value.type,
|
|
32
|
-
});
|
|
33
|
-
});
|
|
34
|
-
}
|
|
35
|
-
fields.push(field);
|
|
36
|
-
});
|
|
37
|
-
return fields;
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
// eslint-disable-next-line class-methods-use-this
|
|
41
|
-
selectUniqueFields(fields) {
|
|
42
|
-
return Array.from(
|
|
43
|
-
new Map(fields.map((item) => [item.name, item])).values(),
|
|
44
|
-
).reduce((uniqueArray, item) => {
|
|
45
|
-
const existingItem = uniqueArray.find(
|
|
46
|
-
(uniqueItem) => uniqueItem.name === item.name,
|
|
47
|
-
);
|
|
48
|
-
if (!existingItem) {
|
|
49
|
-
uniqueArray.push(item);
|
|
50
|
-
} else if (item.required) {
|
|
51
|
-
existingItem.required = true;
|
|
52
|
-
}
|
|
53
|
-
return uniqueArray;
|
|
54
|
-
}, []);
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
groupFieldsFromSchemas(schemas) {
|
|
58
|
-
const result = [];
|
|
59
|
-
schemas.forEach((schema) => {
|
|
60
|
-
const convertedSchema = new ValidateService(this.app, schema).validator;
|
|
61
|
-
|
|
62
|
-
for (const [key, value] of Object.entries(
|
|
63
|
-
convertedSchema?.fieldsInJsonFormat,
|
|
64
|
-
)) {
|
|
65
|
-
result.push({
|
|
66
|
-
name: key,
|
|
67
|
-
type: value.type,
|
|
68
|
-
required: value.required,
|
|
69
|
-
});
|
|
70
|
-
}
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
return result;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
convertDataToDocumentationElement(
|
|
77
|
-
controllerName,
|
|
78
|
-
routesInfo,
|
|
79
|
-
middlewaresInfo,
|
|
80
|
-
routeMiddlewaresReg,
|
|
81
|
-
) {
|
|
82
|
-
return {
|
|
83
|
-
contollerName: controllerName,
|
|
84
|
-
routesInfo: routesInfo.map((route) => {
|
|
85
|
-
const middlewareQueryParams = new ValidateService(
|
|
86
|
-
this.app,
|
|
87
|
-
).getMiddlewareParams(middlewaresInfo, routeMiddlewaresReg, {
|
|
88
|
-
method: route.method.toLowerCase(),
|
|
89
|
-
path: route.fullPath,
|
|
90
|
-
}).query;
|
|
91
|
-
|
|
92
|
-
const middlewareRequestParams = new ValidateService(
|
|
93
|
-
this.app,
|
|
94
|
-
).getMiddlewareParams(middlewaresInfo, routeMiddlewaresReg, {
|
|
95
|
-
method: route.method.toLowerCase(),
|
|
96
|
-
path: route.fullPath,
|
|
97
|
-
}).request;
|
|
98
|
-
|
|
99
|
-
const queryParams = this.groupFieldsFromSchemas(middlewareQueryParams);
|
|
100
|
-
|
|
101
|
-
const requestParams = this.groupFieldsFromSchemas(
|
|
102
|
-
middlewareRequestParams,
|
|
103
|
-
);
|
|
104
|
-
|
|
105
|
-
return {
|
|
106
|
-
[route.fullPath]: {
|
|
107
|
-
method: route.method,
|
|
108
|
-
name: route.name,
|
|
109
|
-
description: route?.description,
|
|
110
|
-
fields: this.selectUniqueFields([
|
|
111
|
-
...this.processingFields(route.fields),
|
|
112
|
-
...requestParams,
|
|
113
|
-
]),
|
|
114
|
-
queryFields: this.selectUniqueFields([
|
|
115
|
-
...this.processingFields(route.queryFields),
|
|
116
|
-
...queryParams,
|
|
117
|
-
]),
|
|
118
|
-
routeMiddlewares: routeMiddlewaresReg
|
|
119
|
-
.map((middleware) => {
|
|
120
|
-
const routeFullPath = route.fullPath.toUpperCase();
|
|
121
|
-
const middlewareFullPath = middleware.fullPath.toUpperCase();
|
|
122
|
-
if (
|
|
123
|
-
route.method.toLowerCase() ===
|
|
124
|
-
middleware.method.toLowerCase() &&
|
|
125
|
-
(middlewareFullPath === routeFullPath ||
|
|
126
|
-
middlewareFullPath === `${routeFullPath}*`)
|
|
127
|
-
) {
|
|
128
|
-
return {
|
|
129
|
-
name: middleware.name,
|
|
130
|
-
params: middleware.params,
|
|
131
|
-
authParams: middleware.authParams,
|
|
132
|
-
};
|
|
133
|
-
}
|
|
134
|
-
return null;
|
|
135
|
-
})
|
|
136
|
-
.filter(Boolean),
|
|
137
|
-
controllerMiddlewares: [
|
|
138
|
-
...new Set(
|
|
139
|
-
middlewaresInfo
|
|
140
|
-
.filter((middleware) => {
|
|
141
|
-
const routeFullPath = route.fullPath.toUpperCase();
|
|
142
|
-
const middlewareFullPath =
|
|
143
|
-
middleware.fullPath.toUpperCase();
|
|
144
|
-
const middlewareFullPathWithSliced = middleware.fullPath
|
|
145
|
-
.toUpperCase()
|
|
146
|
-
.slice(0, -1);
|
|
147
|
-
|
|
148
|
-
return (
|
|
149
|
-
middlewareFullPath === routeFullPath ||
|
|
150
|
-
middlewareFullPath === `${routeFullPath}*` ||
|
|
151
|
-
routeFullPath?.indexOf(middlewareFullPathWithSliced) !==
|
|
152
|
-
-1
|
|
153
|
-
);
|
|
154
|
-
})
|
|
155
|
-
.map(({ name, params, authParams }) => ({
|
|
156
|
-
name,
|
|
157
|
-
params,
|
|
158
|
-
authParams,
|
|
159
|
-
})),
|
|
160
|
-
),
|
|
161
|
-
],
|
|
162
|
-
},
|
|
163
|
-
};
|
|
164
|
-
}),
|
|
165
|
-
};
|
|
166
|
-
}
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
export default DocumentationGenerator;
|
|
@@ -1,89 +0,0 @@
|
|
|
1
|
-
import http from 'node:http';
|
|
2
|
-
// import path from 'node:path';
|
|
3
|
-
// import * as url from 'node:url';
|
|
4
|
-
import express from 'express';
|
|
5
|
-
import RequestLoggerMiddleware from './middleware/RequestLogger.js';
|
|
6
|
-
import I18nMiddleware from './middleware/I18n.js';
|
|
7
|
-
import PrepareAppInfoMiddleware from './middleware/PrepareAppInfo.js';
|
|
8
|
-
import RequestParserMiddleware from './middleware/RequestParser.js';
|
|
9
|
-
import IpDetector from './middleware/IpDetector.js';
|
|
10
|
-
import Cors from './middleware/Cors.js';
|
|
11
|
-
import Base from '../../modules/Base.js';
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
* HTTP server based on Express
|
|
15
|
-
*/
|
|
16
|
-
class HttpServer extends Base {
|
|
17
|
-
constructor(app) {
|
|
18
|
-
super(app);
|
|
19
|
-
this.express = express();
|
|
20
|
-
this.express.disable('x-powered-by');
|
|
21
|
-
|
|
22
|
-
this.express.use(new RequestLoggerMiddleware(this.app).getMiddleware());
|
|
23
|
-
this.express.use(new PrepareAppInfoMiddleware(this.app).getMiddleware());
|
|
24
|
-
this.express.use(new IpDetector(this.app).getMiddleware());
|
|
25
|
-
this.express.use(new I18nMiddleware(this.app).getMiddleware());
|
|
26
|
-
|
|
27
|
-
const httpConfig = this.app.getConfig('http');
|
|
28
|
-
this.express.use(
|
|
29
|
-
new Cors(this.app, {
|
|
30
|
-
origins: httpConfig.corsDomains,
|
|
31
|
-
}).getMiddleware(),
|
|
32
|
-
);
|
|
33
|
-
|
|
34
|
-
this.express.use(new RequestParserMiddleware(this.app).getMiddleware());
|
|
35
|
-
|
|
36
|
-
// As exprress will check numbersof arguments
|
|
37
|
-
// eslint-disable-next-line no-unused-vars
|
|
38
|
-
this.express.use((err, req, res, next) => {
|
|
39
|
-
// error handling
|
|
40
|
-
console.error(err.stack);
|
|
41
|
-
// TODO
|
|
42
|
-
res.status(500).json({ message: 'Something broke!' });
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
this.httpServer = http.createServer(this.express);
|
|
46
|
-
|
|
47
|
-
const listener = this.httpServer.listen(
|
|
48
|
-
httpConfig.port,
|
|
49
|
-
httpConfig.hostname,
|
|
50
|
-
() => {
|
|
51
|
-
const address = listener.address();
|
|
52
|
-
const port = typeof address === 'string' ? 0 : address.port;
|
|
53
|
-
this.logger.info(`App started and listening on port ${port}`);
|
|
54
|
-
if (+port !== +httpConfig.port) {
|
|
55
|
-
// in case we using port 0
|
|
56
|
-
this.app.updateConfig('http', { port });
|
|
57
|
-
this.logger.info(
|
|
58
|
-
`Updating http config to use new port ${
|
|
59
|
-
port
|
|
60
|
-
}. Old was ${httpConfig.port} `,
|
|
61
|
-
);
|
|
62
|
-
}
|
|
63
|
-
},
|
|
64
|
-
);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* Add handle for 404 error
|
|
69
|
-
*/
|
|
70
|
-
add404Page() {
|
|
71
|
-
this.express.use((req, res) => {
|
|
72
|
-
// error handling
|
|
73
|
-
res.status(404).json({ message: '404' });
|
|
74
|
-
});
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
static get loggerGroup() {
|
|
78
|
-
return 'service';
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
/**
|
|
82
|
-
* Stop http server (mostly for unit testing)
|
|
83
|
-
*/
|
|
84
|
-
shutdown() {
|
|
85
|
-
this.httpServer.close();
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
export default HttpServer;
|
|
@@ -1,51 +0,0 @@
|
|
|
1
|
-
import { object } from 'yup';
|
|
2
|
-
import Base from '../../../modules/Base.js';
|
|
3
|
-
|
|
4
|
-
class AbstractMiddleware extends Base {
|
|
5
|
-
constructor(app, params) {
|
|
6
|
-
super(app);
|
|
7
|
-
this.params = params;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
static get description() {
|
|
11
|
-
return 'Middleware description. Please provide own';
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
static get usedAuthParameters() {
|
|
15
|
-
return [];
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
// eslint-disable-next-line class-methods-use-this
|
|
19
|
-
get relatedQueryParameters() {
|
|
20
|
-
// For example yup.object().shape({page: yup.number().required(),limit: yup.number()})
|
|
21
|
-
return object().shape({});
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
// eslint-disable-next-line class-methods-use-this
|
|
25
|
-
get relatedRequestParameters() {
|
|
26
|
-
// For example yup.object().shape({page: yup.number().required(),limit: yup.number()})
|
|
27
|
-
return object().shape({});
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
get relatedReqParameters() {
|
|
31
|
-
return {
|
|
32
|
-
request: this.relatedRequestParameters,
|
|
33
|
-
query: this.relatedQueryParameters,
|
|
34
|
-
};
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
async middleware(req, res, next) {
|
|
38
|
-
this.logger.warn('Middleware is not implemented');
|
|
39
|
-
next();
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
getMiddleware() {
|
|
43
|
-
return this.middleware.bind(this);
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
static get loggerGroup() {
|
|
47
|
-
return 'middleware';
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
export default AbstractMiddleware;
|
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
import AbstractMiddleware from './AbstractMiddleware.js';
|
|
2
|
-
|
|
3
|
-
class AuthMiddleware extends AbstractMiddleware {
|
|
4
|
-
static get description() {
|
|
5
|
-
return 'Allow to pass only if the user provided. Please use any middleware that provide user instance before';
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
async middleware(req, res, next) {
|
|
9
|
-
if (!req.appInfo.user) {
|
|
10
|
-
this.logger.info('User try to access resource without credentials');
|
|
11
|
-
return res.status(401).json({
|
|
12
|
-
error: 'AUTH001',
|
|
13
|
-
message: 'Please login to application',
|
|
14
|
-
});
|
|
15
|
-
}
|
|
16
|
-
return next();
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
export default AuthMiddleware;
|
|
@@ -1,46 +0,0 @@
|
|
|
1
|
-
import AbstractMiddleware from './AbstractMiddleware.js';
|
|
2
|
-
|
|
3
|
-
class Cors extends AbstractMiddleware {
|
|
4
|
-
constructor(app, params) {
|
|
5
|
-
super(app);
|
|
6
|
-
this.params = params;
|
|
7
|
-
if (!Array.isArray(params?.origins) || !params.origins.length) {
|
|
8
|
-
throw new Error('Cors inited without origin config');
|
|
9
|
-
}
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
static get description() {
|
|
13
|
-
return 'Add CORS headers to request';
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
async middleware(req, res, next) {
|
|
17
|
-
for (const host of this.params.origins) {
|
|
18
|
-
if (
|
|
19
|
-
(typeof host === 'string' && req.headers.origin === host) ||
|
|
20
|
-
(host instanceof RegExp && host.test(req.headers.origin))
|
|
21
|
-
) {
|
|
22
|
-
res.set('Access-Control-Allow-Origin', req.headers.origin);
|
|
23
|
-
res.set('Vary', 'Origin');
|
|
24
|
-
|
|
25
|
-
if (req.method === 'OPTIONS') {
|
|
26
|
-
res.set(
|
|
27
|
-
'Access-Control-Allow-Methods',
|
|
28
|
-
'GET,HEAD,PUT,PATCH,POST,DELETE',
|
|
29
|
-
);
|
|
30
|
-
res.set('Vary', 'Origin, Access-Control-Request-Headers');
|
|
31
|
-
|
|
32
|
-
const allowedHeaders = req.headers['access-control-request-headers'];
|
|
33
|
-
if (allowedHeaders) {
|
|
34
|
-
res.set('Access-Control-Allow-Headers', allowedHeaders);
|
|
35
|
-
}
|
|
36
|
-
res.set('Content-Length', '0');
|
|
37
|
-
res.status(204);
|
|
38
|
-
return res.end();
|
|
39
|
-
}
|
|
40
|
-
}
|
|
41
|
-
}
|
|
42
|
-
return next();
|
|
43
|
-
}
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
export default Cors;
|
|
@@ -1,47 +0,0 @@
|
|
|
1
|
-
import AbstractMiddleware from './AbstractMiddleware.js';
|
|
2
|
-
|
|
3
|
-
class GetUserByToken extends AbstractMiddleware {
|
|
4
|
-
static get description() {
|
|
5
|
-
return 'Grab a token and try to parse the user from it. It user exist will add req.appInfo.user variable';
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
// eslint-disable-next-line class-methods-use-this
|
|
9
|
-
get usedAuthParameters() {
|
|
10
|
-
return [
|
|
11
|
-
{
|
|
12
|
-
name: 'Authorization',
|
|
13
|
-
type: 'apiKey',
|
|
14
|
-
in: 'header',
|
|
15
|
-
description: GetUserByToken.description,
|
|
16
|
-
},
|
|
17
|
-
];
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
async middleware(req, res, next) {
|
|
21
|
-
if (req.appInfo.user) {
|
|
22
|
-
this.logger.warn('You call GetUserByToken more then once');
|
|
23
|
-
return next();
|
|
24
|
-
}
|
|
25
|
-
let { token } = req.body || {};
|
|
26
|
-
this.logger.verbose(
|
|
27
|
-
`GetUserByToken token in BODY ${token}. Token in Authorization header ${req.get(
|
|
28
|
-
'Authorization',
|
|
29
|
-
)}`,
|
|
30
|
-
);
|
|
31
|
-
if (!token) {
|
|
32
|
-
token = req.get('Authorization');
|
|
33
|
-
if (!token || token === 'null') {
|
|
34
|
-
// is null express bug*
|
|
35
|
-
return next();
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
const User = this.app.getModel('User');
|
|
39
|
-
const user = await User.getUserByToken(token);
|
|
40
|
-
if (user) {
|
|
41
|
-
req.appInfo.user = user;
|
|
42
|
-
}
|
|
43
|
-
return next();
|
|
44
|
-
}
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
export default GetUserByToken;
|
|
@@ -1,117 +0,0 @@
|
|
|
1
|
-
import i18next from 'i18next';
|
|
2
|
-
import BackendFS from 'i18next-fs-backend';
|
|
3
|
-
import AbstractMiddleware from './AbstractMiddleware.js';
|
|
4
|
-
|
|
5
|
-
class I18n extends AbstractMiddleware {
|
|
6
|
-
cache = {};
|
|
7
|
-
|
|
8
|
-
enabled = true;
|
|
9
|
-
|
|
10
|
-
lookupQuerystring = '';
|
|
11
|
-
|
|
12
|
-
supportedLngs = [];
|
|
13
|
-
|
|
14
|
-
fallbackLng = 'en';
|
|
15
|
-
|
|
16
|
-
/** @type {i18next} */
|
|
17
|
-
i18n = {
|
|
18
|
-
// @ts-ignore
|
|
19
|
-
t: (text) => text,
|
|
20
|
-
language: 'en',
|
|
21
|
-
};
|
|
22
|
-
|
|
23
|
-
constructor(app, params) {
|
|
24
|
-
super(app, params);
|
|
25
|
-
const I18NConfig = this.app.getConfig('i18n');
|
|
26
|
-
|
|
27
|
-
if (I18NConfig.enabled) {
|
|
28
|
-
this.logger.info('Enabling i18n support');
|
|
29
|
-
this.i18n = i18next;
|
|
30
|
-
// eslint-disable-next-line import-x/no-named-as-default-member
|
|
31
|
-
i18next.use(BackendFS).init({
|
|
32
|
-
backend: {
|
|
33
|
-
loadPath: `${this.app.foldersConfig.locales}/{{lng}}/{{ns}}.json`,
|
|
34
|
-
addPath: `${this.app.foldersConfig.locales}/{{lng}}/{{ns}}.missing.json`,
|
|
35
|
-
},
|
|
36
|
-
fallbackLng: I18NConfig.fallbackLng,
|
|
37
|
-
preload: I18NConfig.preload,
|
|
38
|
-
saveMissing: I18NConfig.saveMissing,
|
|
39
|
-
debug: I18NConfig.debug,
|
|
40
|
-
});
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
this.enabled = I18NConfig.enabled;
|
|
44
|
-
this.lookupQuerystring = I18NConfig.lookupQuerystring;
|
|
45
|
-
this.supportedLngs = I18NConfig.supportedLngs;
|
|
46
|
-
this.fallbackLng = I18NConfig.fallbackLng;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
static get description() {
|
|
50
|
-
return 'Provide language detection and translation';
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
async middleware(req, res, next) {
|
|
54
|
-
let i18n;
|
|
55
|
-
|
|
56
|
-
if (this.enabled) {
|
|
57
|
-
let lang = this.detectLang(req);
|
|
58
|
-
if (!lang || this.supportedLngs.indexOf(lang) === -1) {
|
|
59
|
-
this.logger.verbose(
|
|
60
|
-
`Language "${lang}" is not supported or not detected. Using fallback on ${this.fallbackLng}`,
|
|
61
|
-
);
|
|
62
|
-
lang = this.fallbackLng;
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
if (!this.cache[lang]) {
|
|
66
|
-
this.cache[lang] = i18next.cloneInstance({
|
|
67
|
-
initAsync: false,
|
|
68
|
-
lng: lang,
|
|
69
|
-
});
|
|
70
|
-
}
|
|
71
|
-
i18n = this.cache[lang];
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
if (!i18n) {
|
|
75
|
-
i18n = this.i18n;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
req.appInfo.i18n = i18n;
|
|
79
|
-
req.i18n = new Proxy(req.appInfo.i18n, {
|
|
80
|
-
get: (target, prop) => {
|
|
81
|
-
this.logger.warn('Please not use "req.i18n" Use "req.appInfo.i18n"');
|
|
82
|
-
return target[prop];
|
|
83
|
-
},
|
|
84
|
-
});
|
|
85
|
-
|
|
86
|
-
return next();
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
detectors = {
|
|
90
|
-
XLang: (req) => req.get('X-Lang'), // grab from header
|
|
91
|
-
query: (req) => (req.query ? req.query[this.lookupQuerystring] : false), // grab from query
|
|
92
|
-
user: (req) => req.appInfo?.user?.locale, // what if we have a user and user have a defined locale?
|
|
93
|
-
};
|
|
94
|
-
|
|
95
|
-
detectorOrder = ['XLang', 'query', 'user'];
|
|
96
|
-
|
|
97
|
-
detectLang(req, isUseShortCode = true) {
|
|
98
|
-
let lang = '';
|
|
99
|
-
for (const detectorName of this.detectorOrder) {
|
|
100
|
-
const lng = this.detectors[detectorName](req);
|
|
101
|
-
if (!lng) {
|
|
102
|
-
continue;
|
|
103
|
-
}
|
|
104
|
-
if (i18next.services.languageUtils.isSupportedCode(lng)) {
|
|
105
|
-
if (isUseShortCode) {
|
|
106
|
-
lang = i18next.services.languageUtils.getLanguagePartFromCode(lng);
|
|
107
|
-
} else {
|
|
108
|
-
lang = lng;
|
|
109
|
-
}
|
|
110
|
-
break;
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
return lang;
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
export default I18n;
|