@aristid/leav-types 1.4.0 → 1.4.1-9e3f8604

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.
Files changed (25) hide show
  1. package/apps/core/config/default.d.ts +6 -0
  2. package/apps/core/src/__tests__/e2e/api/permissions/extendedLibraryPermissions.test.d.ts +1 -0
  3. package/apps/core/src/__tests__/integration/infra/ramService.test.d.ts +1 -0
  4. package/apps/core/src/_types/config.d.ts +8 -0
  5. package/apps/core/src/_types/elasticSearch.d.ts +2 -0
  6. package/apps/core/src/_types/errors.d.ts +1 -0
  7. package/apps/core/src/app/auth/authApp.d.ts +3 -1
  8. package/apps/core/src/app/core/index.d.ts +1 -0
  9. package/apps/core/src/app/core/logsCollectorApp.d.ts +9 -0
  10. package/apps/core/src/domain/logsCollector/index.d.ts +1 -0
  11. package/apps/core/src/domain/logsCollector/logsCollectorDomain.d.ts +16 -0
  12. package/apps/core/src/domain/permission/helpers/index.d.ts +1 -1
  13. package/apps/core/src/domain/permission/helpers/recordInCreationBypass.d.ts +8 -3
  14. package/apps/core/src/domain/permission/recordAttributePermissionDomain.d.ts +2 -2
  15. package/apps/core/src/domain/permission/recordPermissionDomain.d.ts +2 -2
  16. package/apps/core/src/domain/record/helpers/getAccessPermissionFilters.d.ts +6 -5
  17. package/apps/core/src/domain/record/recordDomain.d.ts +1 -0
  18. package/apps/core/src/domain/value/valueDomain.d.ts +3 -1
  19. package/apps/core/src/infra/cache/ramService.d.ts +2 -2
  20. package/apps/core/src/infra/elasticSearch/elasticSearchService.d.ts +5 -4
  21. package/apps/core/src/interface/index.d.ts +1 -0
  22. package/apps/core/src/interface/logsCollector.d.ts +9 -0
  23. package/libs/logger/src/locationInfoFormatters.d.ts +10 -0
  24. package/libs/logger/src/logger.d.ts +1 -1
  25. package/package.json +2 -2
@@ -3,6 +3,7 @@ export declare let coreMode: string;
3
3
  export declare namespace server {
4
4
  let host: string;
5
5
  let port: number;
6
+ let keepAliveTimeout: number;
6
7
  let publicUrl: string;
7
8
  let basePath: string;
8
9
  let allowIntrospection: boolean;
@@ -214,6 +215,11 @@ export declare namespace elasticSearch {
214
215
  export let indexPrefix: string;
215
216
  let url_2: string;
216
217
  export { url_2 as url };
218
+ export let ilmPolicyName: string;
219
+ export let templateName: string;
220
+ }
221
+ export declare namespace logsCollector {
222
+ let queue: string;
217
223
  }
218
224
  export declare let pluginsPath: string | any[];
219
225
  export { _export as export, _import as import };
@@ -30,6 +30,7 @@ export interface IConfig {
30
30
  dbProfiler: IDbProfilerConfig;
31
31
  instanceId: string;
32
32
  elasticSearch: IElasticSearchConfig;
33
+ logsCollector: ILogsCollector;
33
34
  pluginsPath: string[];
34
35
  bugsnag: IBugsnag;
35
36
  matomo: IMatomo;
@@ -41,11 +42,13 @@ export declare enum CoreMode {
41
42
  INDEXATION_MANAGER = "indexationManager",
42
43
  TASKS_MANAGER_MASTER = "tasksManager:master",
43
44
  TASKS_MANAGER_WORKER = "tasksManager:worker",
45
+ LOGS_COLLECTOR = "logsCollector",
44
46
  CLI = "cli"
45
47
  }
46
48
  export interface IServer {
47
49
  host: string;
48
50
  port: number;
51
+ keepAliveTimeout: number;
49
52
  publicUrl: string;
50
53
  basePath: string;
51
54
  allowIntrospection: boolean;
@@ -225,6 +228,8 @@ export interface IDbProfilerConfig {
225
228
  export interface IElasticSearchConfig {
226
229
  indexPrefix: string;
227
230
  url: string;
231
+ ilmPolicyName: string;
232
+ templateName: string;
228
233
  }
229
234
  export interface IBugsnag {
230
235
  enable: boolean;
@@ -238,3 +243,6 @@ export interface IMatomo {
238
243
  url?: string;
239
244
  siteId?: string;
240
245
  }
246
+ export interface ILogsCollector {
247
+ queue: string;
248
+ }
@@ -0,0 +1,2 @@
1
+ import { type IDbEvent, type IDbPayload } from '@leav/utils';
2
+ export type WritableMessage = Omit<IDbEvent, 'payload' | 'emitter'> & IDbPayload;
@@ -4,6 +4,7 @@ export declare enum ErrorTypes {
4
4
  INTERNAL_ERROR = "INTERNAL_ERROR"
5
5
  }
6
6
  export declare enum Errors {
7
+ ATTRIBUTE_USED_BY_LIBRARY = "ATTRIBUTE_USED_BY_LIBRARY",
7
8
  ATTRIBUTE_USED_IN_METADATA = "ATTRIBUTE_USED_IN_METADATA",
8
9
  CANNOT_SAVE_METADATA = "CANNOT_SAVE_METADATA",
9
10
  DUPLICATE_DIRECTORY_NAMES = "DUPLICATE_DIRECTORY_NAMES",
@@ -15,6 +15,7 @@ import { type IncomingHttpHeaders } from 'http';
15
15
  import { type IRecordRepo } from '../../infra/record/recordRepo';
16
16
  import { type IGraphqlAppModule } from 'app/graphql/graphqlApp';
17
17
  import { type IServerRouteAppModule } from 'interface/server';
18
+ import { type GetSystemQueryContext } from 'utils/helpers/getSystemQueryContext';
18
19
  export interface IAuthApp extends IGraphqlAppModule, IServerRouteAppModule {
19
20
  validateRequestToken(params: {
20
21
  apiKey?: string;
@@ -34,6 +35,7 @@ export interface IAuthAppDeps {
34
35
  'core.infra.oidc.oidcClientService': IOIDCClientService;
35
36
  'core.app.helpers.initQueryContext': InitQueryContextFunc;
36
37
  'core.app.helpers.convertOIDCIdentifier': IConvertOIDCIdentifier;
38
+ 'core.utils.getSystemQueryContext': GetSystemQueryContext;
37
39
  config: IConfig;
38
40
  }
39
- export default function ({ 'core.domain.value': valueDomain, 'core.domain.record': recordDomain, 'core.infra.record': recordRepo, 'core.domain.apiKey': apiKeyDomain, 'core.domain.user': userDomain, 'core.utils.logger': logger, 'core.infra.cache.cacheService': cacheService, 'core.infra.oidc.oidcClientService': oidcClientService, 'core.app.helpers.initQueryContext': initQueryContext, 'core.app.helpers.convertOIDCIdentifier': convertOIDCIdentifier, config }: IAuthAppDeps): IAuthApp;
41
+ export default function ({ 'core.domain.value': valueDomain, 'core.domain.record': recordDomain, 'core.infra.record': recordRepo, 'core.domain.apiKey': apiKeyDomain, 'core.domain.user': userDomain, 'core.utils.logger': logger, 'core.infra.cache.cacheService': cacheService, 'core.infra.oidc.oidcClientService': oidcClientService, 'core.app.helpers.initQueryContext': initQueryContext, 'core.app.helpers.convertOIDCIdentifier': convertOIDCIdentifier, 'core.utils.getSystemQueryContext': getSystemQueryContext, config }: IAuthAppDeps): IAuthApp;
@@ -10,6 +10,7 @@ export { default as globalSettings } from './globalSettingsApp';
10
10
  export { default as subscriptionsHelper } from './helpers/subscriptions';
11
11
  export { default as import } from './importApp';
12
12
  export { default as indexationManager } from './indexationManagerApp';
13
+ export { default as logsCollector } from './logsCollectorApp';
13
14
  export { default as library } from './libraryApp/libraryApp';
14
15
  export { default as log } from './logApp';
15
16
  export { default as permission } from './permissionApp/permissionApp';
@@ -0,0 +1,9 @@
1
+ import { type ILogsCollectorDomain } from 'domain/logsCollector/logsCollectorDomain';
2
+ export interface ILogsCollectorApp {
3
+ init(): Promise<void>;
4
+ }
5
+ interface IDeps {
6
+ 'core.domain.logsCollector': ILogsCollectorDomain;
7
+ }
8
+ export default function ({ 'core.domain.logsCollector': logsCollector }: IDeps): ILogsCollectorApp;
9
+ export {};
@@ -0,0 +1 @@
1
+ export { default } from './logsCollectorDomain';
@@ -0,0 +1,16 @@
1
+ import { type IAmqpService } from '@leav/message-broker';
2
+ import { type ILogger } from '@leav/logger';
3
+ import type * as Config from '_types/config';
4
+ import { type IIndexationService } from '../../infra/indexation/indexationService';
5
+ import { type IElasticSearchService } from '../../infra/elasticSearch/elasticSearchService';
6
+ export interface ILogsCollectorDomain {
7
+ init(): Promise<void>;
8
+ }
9
+ export interface ILogsCollectorDomainDeps {
10
+ config: Config.IConfig;
11
+ 'core.infra.amqpService': IAmqpService;
12
+ 'core.infra.indexation.indexationService': IIndexationService;
13
+ 'core.utils.logger': ILogger;
14
+ 'core.infra.elasticSearch.service': IElasticSearchService;
15
+ }
16
+ export default function ({ config, 'core.infra.amqpService': amqpService, 'core.infra.indexation.indexationService': indexationService, 'core.utils.logger': logger, 'core.infra.elasticSearch.service': esService }: ILogsCollectorDomainDeps): ILogsCollectorDomain;
@@ -5,4 +5,4 @@ export { default as permissionsByActions } from './permissionsByActions';
5
5
  export { default as reducePermissionsArray } from './reducePermissionsArray';
6
6
  export { default as simplePermission } from './simplePermission';
7
7
  export { default as treeBasedPermissions } from './treeBasedPermissions';
8
- export { default as recordInCreationByPass } from './recordInCreationBypass';
8
+ export { default as recordInCreationBypass } from './recordInCreationBypass';
@@ -1,6 +1,11 @@
1
1
  import { type IRecord } from '../../../_types/record';
2
2
  import { type IQueryInfos } from '../../../_types/queryInfos';
3
- export interface IRecordInCreationByPassHelper {
4
- recordInCreationByPass: (record: IRecord, ctx: IQueryInfos) => boolean;
3
+ import { type IRecordRepo } from '../../../infra/record/recordRepo';
4
+ export interface IRecordInCreationBypassHelper {
5
+ recordInCreationBypass: (record: IRecord, ctx: IQueryInfos) => boolean;
6
+ recordInCreationBypassById: (libraryId: string, recordId: string, ctx: IQueryInfos) => Promise<boolean>;
5
7
  }
6
- export default function (): IRecordInCreationByPassHelper;
8
+ export interface IRecordInCreationBypassHelperDeps {
9
+ 'core.infra.record': IRecordRepo;
10
+ }
11
+ export default function ({ 'core.infra.record': recordRepo }: IRecordInCreationBypassHelperDeps): IRecordInCreationBypassHelper;
@@ -8,7 +8,7 @@ import { type IPermissionByUserGroupsHelper } from './helpers/permissionByUserGr
8
8
  import { type ITreeBasedPermissionHelper } from './helpers/treeBasedPermissions';
9
9
  import { type IGetRecordAttributeHeritedPermissionsParams as IGetRecordAttributeInheritedPermissionsParams } from './_types';
10
10
  import { type IRecordRepo } from '../../infra/record/recordRepo';
11
- import { type IRecordInCreationByPassHelper } from './helpers/recordInCreationBypass';
11
+ import { type IRecordInCreationBypassHelper } from './helpers/recordInCreationBypass';
12
12
  export interface IRecordAttributePermissionDomain {
13
13
  getRecordAttributePermission(action: RecordAttributePermissionsActions, userGroupId: string, attributeId: string, recordLibrary: string, recordId: string, ctx: IQueryInfos): Promise<boolean>;
14
14
  getInheritedRecordAttributePermission(params: IGetRecordAttributeInheritedPermissionsParams, ctx: IQueryInfos): Promise<boolean>;
@@ -18,7 +18,7 @@ export interface IRecordAttributePermissionDomainDeps {
18
18
  'core.domain.permission.helpers.treeBasedPermissions': ITreeBasedPermissionHelper;
19
19
  'core.domain.permission.helpers.permissionByUserGroups': IPermissionByUserGroupsHelper;
20
20
  'core.domain.permission.helpers.defaultPermission': IDefaultPermissionHelper;
21
- 'core.domain.permission.helpers.recordInCreationByPass': IRecordInCreationByPassHelper;
21
+ 'core.domain.permission.helpers.recordInCreationBypass': IRecordInCreationBypassHelper;
22
22
  'core.domain.attribute': IAttributeDomain;
23
23
  'core.infra.value': IValueRepo;
24
24
  'core.infra.record': IRecordRepo;
@@ -8,7 +8,7 @@ import { type ILibraryPermissionDomain } from './libraryPermissionDomain';
8
8
  import { type IEstimateTreeValueRecordPermissionParams, type IGetInheritedRecordPermissionParams, type IGetRecordPermissionParams } from './_types';
9
9
  import { type ITreeRepo } from '../../infra/tree/treeRepo';
10
10
  import { type IRecordRepo } from '../../infra/record/recordRepo';
11
- import { type IRecordInCreationByPassHelper } from './helpers/recordInCreationBypass';
11
+ import { type IRecordInCreationBypassHelper } from './helpers/recordInCreationBypass';
12
12
  export interface IRecordPermissionDomain {
13
13
  getRecordPermission(params: IGetRecordPermissionParams): Promise<boolean>;
14
14
  getInheritedRecordPermission(params: IGetInheritedRecordPermissionParams): Promise<boolean>;
@@ -19,7 +19,7 @@ export interface IRecordPermissionDomainDeps {
19
19
  'core.domain.permission.helpers.treeBasedPermissions': ITreeBasedPermissionHelper;
20
20
  'core.domain.permission.helpers.permissionByUserGroups': IPermissionByUserGroupsHelper;
21
21
  'core.domain.permission.helpers.defaultPermission': IDefaultPermissionHelper;
22
- 'core.domain.permission.helpers.recordInCreationByPass': IRecordInCreationByPassHelper;
22
+ 'core.domain.permission.helpers.recordInCreationBypass': IRecordInCreationBypassHelper;
23
23
  'core.domain.attribute': IAttributeDomain;
24
24
  'core.domain.helpers.getCoreEntityById': GetCoreEntityByIdFunc;
25
25
  'core.infra.value': IValueRepo;
@@ -11,14 +11,15 @@ interface IAccessPermissionFilterDeps {
11
11
  'core.infra.permission': IPermissionRepo;
12
12
  'core.domain.permission.helpers.defaultPermission': IDefaultPermissionHelper;
13
13
  }
14
+ interface INodeIdsByPermissions {
15
+ true: Array<ITreeNode['id']>;
16
+ false: Array<ITreeNode['id']>;
17
+ }
14
18
  export interface IGetAccessPermissionsValue {
15
19
  treeId: string;
16
20
  attribute: IAttribute;
17
- permissions: {
18
- true: Array<ITreeNode['id']>;
19
- false: Array<ITreeNode['id']>;
20
- };
21
+ permissions: INodeIdsByPermissions;
21
22
  }
22
- export type IGetAccessPermissions = (groupsIds: string[][], library: string, deps: IAccessPermissionFilterDeps, ctx: IQueryInfos) => Promise<IGetAccessPermissionsValue[]>;
23
+ export type IGetAccessPermissions = (groupsIds: string[][], library: string, existingFiltersOnTreeIds: string[], deps: IAccessPermissionFilterDeps, ctx: IQueryInfos) => Promise<IGetAccessPermissionsValue[]>;
23
24
  declare const getAccessPermissionsFilters: IGetAccessPermissions;
24
25
  export default getAccessPermissionsFilters;
@@ -26,6 +26,7 @@ import { type DeleteRecordHelper } from './helpers/deleteRecord';
26
26
  import { type CreateRecordHelper } from './helpers/createRecord';
27
27
  import { type IElementAncestorsHelper } from 'domain/tree/helpers/elementAncestors';
28
28
  import { type ILogger } from '@leav/logger';
29
+ export declare const ATTRIBUTE_ACTIVE = "active";
29
30
  export interface IRecordDomain {
30
31
  /**
31
32
  * Create empty record
@@ -24,6 +24,7 @@ import { type IDeleteValueParams, type IRunActionListParams } from './_types';
24
24
  import { type DeleteRecordHelper } from 'domain/record/helpers/deleteRecord';
25
25
  import { type CreateRecordHelper } from 'domain/record/helpers/createRecord';
26
26
  import { type IfLibraryJoinLinkAttribute } from '../attribute/helpers/ifLibraryJoinLinkAttribute';
27
+ import { type IRecordInCreationBypassHelper } from '../permission/helpers/recordInCreationBypass';
27
28
  export interface ISaveBatchValueError {
28
29
  type: string;
29
30
  message: string;
@@ -107,6 +108,7 @@ export interface IValueDomainDeps {
107
108
  'core.domain.record.helpers.sendRecordUpdateEvent': SendRecordUpdateEventHelper;
108
109
  'core.domain.record.helpers.createRecord': CreateRecordHelper;
109
110
  'core.domain.record.helpers.deleteRecord': DeleteRecordHelper;
111
+ 'core.domain.permission.helpers.recordInCreationBypass': IRecordInCreationBypassHelper;
110
112
  'core.domain.attribute.helpers.ifLibraryJoinLinkAttribute': IfLibraryJoinLinkAttribute;
111
113
  'core.domain.versionProfile': IVersionProfileDomain;
112
114
  'core.infra.record': IRecordRepo;
@@ -116,5 +118,5 @@ export interface IValueDomainDeps {
116
118
  'core.utils.logger': ILogger;
117
119
  'core.domain.tree': ITreeDomain;
118
120
  }
119
- declare const valueDomain: ({ config, "core.domain.actionsList": actionsListDomain, "core.domain.attribute": attributeDomain, "core.domain.permission.recordAttribute": recordAttributePermissionDomain, "core.domain.permission.record": recordPermissionDomain, "core.domain.eventsManager": eventsManager, "core.domain.helpers.validate": validate, "core.domain.helpers.updateRecordLastModif": updateRecordLastModif, "core.domain.tree.helpers.elementAncestors": elementAncestors, "core.domain.tree.helpers.getDefaultElement": getDefaultElementHelper, "core.domain.record.helpers.sendRecordUpdateEvent": sendRecordUpdateEvent, "core.domain.record.helpers.createRecord": createRecordHelper, "core.domain.record.helpers.deleteRecord": deleteRecordHelper, "core.domain.attribute.helpers.ifLibraryJoinLinkAttribute": ifLibraryJoinLinkAttribute, "core.domain.versionProfile": versionProfileDomain, "core.infra.record": recordRepo, "core.infra.tree": treeRepo, "core.infra.value": valueRepo, "core.utils": utils, "core.utils.logger": logger }: IValueDomainDeps) => IValueDomain;
121
+ declare const valueDomain: ({ config, "core.domain.actionsList": actionsListDomain, "core.domain.attribute": attributeDomain, "core.domain.permission.recordAttribute": recordAttributePermissionDomain, "core.domain.permission.record": recordPermissionDomain, "core.domain.eventsManager": eventsManager, "core.domain.helpers.validate": validate, "core.domain.helpers.updateRecordLastModif": updateRecordLastModif, "core.domain.tree.helpers.elementAncestors": elementAncestors, "core.domain.tree.helpers.getDefaultElement": getDefaultElementHelper, "core.domain.record.helpers.sendRecordUpdateEvent": sendRecordUpdateEvent, "core.domain.record.helpers.createRecord": createRecordHelper, "core.domain.permission.helpers.recordInCreationBypass": recordInCreationBypassHelper, "core.domain.record.helpers.deleteRecord": deleteRecordHelper, "core.domain.attribute.helpers.ifLibraryJoinLinkAttribute": ifLibraryJoinLinkAttribute, "core.domain.versionProfile": versionProfileDomain, "core.infra.record": recordRepo, "core.infra.tree": treeRepo, "core.infra.value": valueRepo, "core.utils": utils, "core.utils.logger": logger }: IValueDomainDeps) => IValueDomain;
120
122
  export default valueDomain;
@@ -1,9 +1,9 @@
1
- import { type IConfig } from '_types/config';
2
1
  import { type ICacheService } from './cacheService';
3
2
  import { type RedisClientType } from './redis';
3
+ import { type IConfig } from '../../_types/config';
4
4
  interface IDeps {
5
5
  config?: IConfig;
6
6
  'core.infra.redis'?: RedisClientType;
7
7
  }
8
- export default function ({ config, 'core.infra.redis': redis }: IDeps): ICacheService;
8
+ export default function ({ 'core.infra.redis': redis, config }: IDeps): ICacheService;
9
9
  export {};
@@ -1,6 +1,6 @@
1
1
  import { Client, type estypes } from '@elastic/elasticsearch';
2
2
  import { type IConfig } from '_types/config';
3
- import { type IQueryInfos } from '_types/queryInfos';
3
+ import { type Log } from '@leav/utils';
4
4
  export interface IElasticsearchServiceSearchResponse<T> {
5
5
  total: number;
6
6
  hits: T[];
@@ -17,11 +17,12 @@ interface IElasticsearchServiceSearchParams {
17
17
  }
18
18
  export interface IElasticSearchService {
19
19
  client: Client;
20
- search: <T>(params: IElasticsearchServiceSearchParams, ctx: IQueryInfos) => Promise<IElasticsearchServiceSearchResponse<T>>;
20
+ search: <T>(params: IElasticsearchServiceSearchParams) => Promise<IElasticsearchServiceSearchResponse<T>>;
21
+ writeData: (indexName: string, data: Log) => Promise<void>;
21
22
  }
22
- interface IDeps {
23
+ export interface IElasticSearchServiceDeps {
23
24
  'core.infra.elasticSearch.client'?: Client;
24
25
  config?: IConfig;
25
26
  }
26
- export default function ({ config }: IDeps): IElasticSearchService;
27
+ export default function ({ config }: IElasticSearchServiceDeps): IElasticSearchService;
27
28
  export {};
@@ -1,5 +1,6 @@
1
1
  export { default as cli } from './cli';
2
2
  export { default as filesManager } from './filesManager';
3
3
  export { default as indexationManager } from './indexationManager';
4
+ export { default as logsCollector } from './logsCollector';
4
5
  export { default as tasksManager } from './tasksManager';
5
6
  export { default as server } from './server';
@@ -0,0 +1,9 @@
1
+ import { type ILogsCollectorApp } from 'app/core/logsCollectorApp';
2
+ export interface ILogsCollectorInterface {
3
+ init(): Promise<void>;
4
+ }
5
+ interface IDeps {
6
+ 'core.app.core.logsCollector': ILogsCollectorApp;
7
+ }
8
+ export default function ({ 'core.app.core.logsCollector': logsCollector }: IDeps): ILogsCollectorInterface;
9
+ export {};
@@ -0,0 +1,10 @@
1
+ import winston from 'winston';
2
+ interface ICallerInfo {
3
+ path: string;
4
+ line: string;
5
+ col: string;
6
+ }
7
+ export declare function getLocationInfo(): ICallerInfo | null;
8
+ export declare const addLocationInfoInLog: winston.Logform.FormatWrap;
9
+ export declare const mergeLocationInfoInLog: winston.Logform.FormatWrap;
10
+ export {};
@@ -1,4 +1,4 @@
1
- import * as winston from 'winston';
1
+ import winston from 'winston';
2
2
  import { type ILoggerConfig } from './config';
3
3
  export type ILogger = Pick<typeof winston, 'error' | 'warn' | 'info' | 'log' | 'verbose' | 'debug' | 'silly'>;
4
4
  export declare function configureLogger(config: ILoggerConfig): void;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@aristid/leav-types",
3
- "version": "1.4.0",
3
+ "version": "1.4.1-9e3f8604",
4
4
  "description": "Shared Leav types",
5
5
  "scripts": {
6
6
  "tscheck": "",
@@ -14,7 +14,7 @@
14
14
  "license": "LGPL3",
15
15
  "repository": "https://github.com/leav-solutions/leav-engine",
16
16
  "dependencies": {
17
- "@leav/utils": "1.4.0",
17
+ "@leav/utils": "1.4.1",
18
18
  "@types/amqplib": "0.10.7",
19
19
  "@types/express": "5.0.0",
20
20
  "@types/jest": "29.5.14",