@ruiapp/rapid-core 0.3.0 → 0.3.2

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 (101) hide show
  1. package/CHANGELOG.md +7 -7
  2. package/dist/index.d.ts +2 -0
  3. package/dist/index.js +359 -134
  4. package/dist/plugins/license/LicensePlugin.d.ts +23 -0
  5. package/dist/plugins/license/LicensePluginTypes.d.ts +78 -0
  6. package/dist/plugins/license/LicenseService.d.ts +22 -0
  7. package/dist/plugins/license/actionHandlers/getLicense.d.ts +6 -0
  8. package/dist/plugins/license/actionHandlers/index.d.ts +3 -0
  9. package/dist/plugins/license/helpers/certHelper.d.ts +2 -0
  10. package/dist/plugins/license/helpers/cryptoHelper.d.ts +8 -0
  11. package/dist/plugins/license/models/index.d.ts +2 -0
  12. package/dist/plugins/license/routes/getLicense.d.ts +12 -0
  13. package/dist/plugins/license/routes/index.d.ts +12 -0
  14. package/dist/utilities/typeUtility.d.ts +1 -0
  15. package/package.json +1 -1
  16. package/src/bootstrapApplicationConfig.ts +631 -631
  17. package/src/core/response.ts +76 -76
  18. package/src/core/routeContext.ts +47 -47
  19. package/src/core/server.ts +142 -142
  20. package/src/dataAccess/columnTypeMapper.ts +22 -22
  21. package/src/dataAccess/dataAccessTypes.ts +163 -163
  22. package/src/dataAccess/dataAccessor.ts +135 -135
  23. package/src/dataAccess/entityManager.ts +1718 -1718
  24. package/src/dataAccess/entityMapper.ts +100 -100
  25. package/src/dataAccess/propertyMapper.ts +28 -28
  26. package/src/deno-std/http/cookie.ts +372 -372
  27. package/src/helpers/entityHelpers.ts +76 -76
  28. package/src/helpers/filterHelper.ts +148 -148
  29. package/src/helpers/metaHelper.ts +89 -89
  30. package/src/helpers/runCollectionEntityActionHandler.ts +27 -27
  31. package/src/index.ts +57 -54
  32. package/src/plugins/auth/AuthPlugin.ts +85 -85
  33. package/src/plugins/auth/actionHandlers/changePassword.ts +54 -54
  34. package/src/plugins/auth/actionHandlers/createSession.ts +75 -63
  35. package/src/plugins/auth/actionHandlers/resetPassword.ts +38 -38
  36. package/src/plugins/cronJob/CronJobPlugin.ts +112 -112
  37. package/src/plugins/dataManage/DataManagePlugin.ts +163 -163
  38. package/src/plugins/dataManage/actionHandlers/addEntityRelations.ts +20 -20
  39. package/src/plugins/dataManage/actionHandlers/countCollectionEntities.ts +16 -16
  40. package/src/plugins/dataManage/actionHandlers/createCollectionEntitiesBatch.ts +42 -42
  41. package/src/plugins/dataManage/actionHandlers/createCollectionEntity.ts +24 -24
  42. package/src/plugins/dataManage/actionHandlers/deleteCollectionEntities.ts +38 -38
  43. package/src/plugins/dataManage/actionHandlers/deleteCollectionEntityById.ts +22 -22
  44. package/src/plugins/dataManage/actionHandlers/findCollectionEntities.ts +26 -26
  45. package/src/plugins/dataManage/actionHandlers/findCollectionEntityById.ts +21 -21
  46. package/src/plugins/dataManage/actionHandlers/removeEntityRelations.ts +20 -20
  47. package/src/plugins/dataManage/actionHandlers/updateCollectionEntityById.ts +41 -41
  48. package/src/plugins/entityAccessControl/EntityAccessControlPlugin.ts +146 -146
  49. package/src/plugins/fileManage/FileManagePlugin.ts +52 -52
  50. package/src/plugins/fileManage/actionHandlers/downloadDocument.ts +65 -65
  51. package/src/plugins/fileManage/actionHandlers/downloadFile.ts +44 -44
  52. package/src/plugins/license/LicensePlugin.ts +79 -0
  53. package/src/plugins/license/LicensePluginTypes.ts +95 -0
  54. package/src/plugins/license/LicenseService.ts +112 -0
  55. package/src/plugins/license/actionHandlers/getLicense.ts +18 -0
  56. package/src/plugins/license/actionHandlers/index.ts +4 -0
  57. package/src/plugins/license/helpers/certHelper.ts +21 -0
  58. package/src/plugins/license/helpers/cryptoHelper.ts +47 -0
  59. package/src/plugins/license/models/index.ts +1 -0
  60. package/src/plugins/license/routes/getLicense.ts +15 -0
  61. package/src/plugins/license/routes/index.ts +3 -0
  62. package/src/plugins/mail/MailPlugin.ts +74 -74
  63. package/src/plugins/mail/MailPluginTypes.ts +27 -27
  64. package/src/plugins/mail/MailService.ts +38 -38
  65. package/src/plugins/mail/actionHandlers/index.ts +3 -3
  66. package/src/plugins/mail/models/index.ts +1 -1
  67. package/src/plugins/mail/routes/index.ts +1 -1
  68. package/src/plugins/metaManage/MetaManagePlugin.ts +504 -504
  69. package/src/plugins/notification/NotificationPlugin.ts +68 -68
  70. package/src/plugins/notification/NotificationPluginTypes.ts +13 -13
  71. package/src/plugins/notification/NotificationService.ts +25 -25
  72. package/src/plugins/notification/actionHandlers/index.ts +3 -3
  73. package/src/plugins/notification/models/Notification.ts +60 -57
  74. package/src/plugins/notification/models/index.ts +3 -3
  75. package/src/plugins/notification/routes/index.ts +1 -1
  76. package/src/plugins/routeManage/RouteManagePlugin.ts +62 -62
  77. package/src/plugins/sequence/SequencePlugin.ts +136 -136
  78. package/src/plugins/sequence/SequencePluginTypes.ts +69 -69
  79. package/src/plugins/sequence/SequenceService.ts +81 -81
  80. package/src/plugins/sequence/actionHandlers/generateSn.ts +32 -32
  81. package/src/plugins/sequence/segments/autoIncrement.ts +78 -78
  82. package/src/plugins/sequence/segments/dayOfMonth.ts +17 -17
  83. package/src/plugins/sequence/segments/literal.ts +14 -14
  84. package/src/plugins/sequence/segments/month.ts +17 -17
  85. package/src/plugins/sequence/segments/parameter.ts +18 -18
  86. package/src/plugins/sequence/segments/year.ts +17 -17
  87. package/src/plugins/setting/SettingPlugin.ts +68 -68
  88. package/src/plugins/setting/SettingPluginTypes.ts +37 -37
  89. package/src/plugins/setting/models/SystemSettingItem.ts +48 -42
  90. package/src/plugins/setting/models/UserSettingItem.ts +55 -49
  91. package/src/plugins/stateMachine/StateMachinePlugin.ts +186 -186
  92. package/src/plugins/stateMachine/StateMachinePluginTypes.ts +48 -48
  93. package/src/plugins/stateMachine/actionHandlers/sendStateMachineEvent.ts +51 -51
  94. package/src/plugins/webhooks/WebhooksPlugin.ts +148 -148
  95. package/src/plugins/webhooks/pluginConfig.ts +75 -75
  96. package/src/queryBuilder/queryBuilder.ts +665 -665
  97. package/src/server.ts +463 -463
  98. package/src/types.ts +701 -701
  99. package/src/utilities/errorUtility.ts +15 -15
  100. package/src/utilities/pathUtility.ts +14 -14
  101. package/src/utilities/typeUtility.ts +15 -11
@@ -1,51 +1,51 @@
1
- import { ActionHandlerContext } from "~/core/actionHandler";
2
- import { RapidPlugin } from "~/core/server";
3
- import { SendStateMachineEventInput, SendStateMachineEventOptions } from "../StateMachinePluginTypes";
4
- import { getStateMachineNextSnapshot } from "../stateMachineHelper";
5
-
6
- export const code = "sendStateMachineEvent";
7
-
8
- export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: SendStateMachineEventOptions) {
9
- const { server, routerContext } = ctx;
10
- const { response } = routerContext;
11
-
12
- const input: SendStateMachineEventInput = ctx.input;
13
- if (options?.code) {
14
- input.code = options.code;
15
- }
16
-
17
- if (!input.code) {
18
- throw new Error(`State machine code is required when sending event.`);
19
- }
20
-
21
- const stateMachineDataAccessor = server.getDataAccessor({
22
- singularCode: "state_machine",
23
- });
24
-
25
- const stateMachine = await stateMachineDataAccessor.findOne({
26
- filters: [
27
- {
28
- operator: "eq",
29
- field: "code",
30
- value: input.code,
31
- },
32
- ],
33
- });
34
-
35
- if (!stateMachine) {
36
- throw new Error(`State machine with code '${input.code}' was not found.`);
37
- }
38
-
39
- stateMachine.config.id = input.code;
40
-
41
- const snapshot = await getStateMachineNextSnapshot({
42
- machineConfig: stateMachine.config,
43
- context: input.context,
44
- currentState: input.currentState,
45
- event: input.event,
46
- });
47
-
48
- response.json({
49
- state: snapshot.value,
50
- });
51
- }
1
+ import { ActionHandlerContext } from "~/core/actionHandler";
2
+ import { RapidPlugin } from "~/core/server";
3
+ import { SendStateMachineEventInput, SendStateMachineEventOptions } from "../StateMachinePluginTypes";
4
+ import { getStateMachineNextSnapshot } from "../stateMachineHelper";
5
+
6
+ export const code = "sendStateMachineEvent";
7
+
8
+ export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: SendStateMachineEventOptions) {
9
+ const { server, routerContext } = ctx;
10
+ const { response } = routerContext;
11
+
12
+ const input: SendStateMachineEventInput = ctx.input;
13
+ if (options?.code) {
14
+ input.code = options.code;
15
+ }
16
+
17
+ if (!input.code) {
18
+ throw new Error(`State machine code is required when sending event.`);
19
+ }
20
+
21
+ const stateMachineDataAccessor = server.getDataAccessor({
22
+ singularCode: "state_machine",
23
+ });
24
+
25
+ const stateMachine = await stateMachineDataAccessor.findOne({
26
+ filters: [
27
+ {
28
+ operator: "eq",
29
+ field: "code",
30
+ value: input.code,
31
+ },
32
+ ],
33
+ });
34
+
35
+ if (!stateMachine) {
36
+ throw new Error(`State machine with code '${input.code}' was not found.`);
37
+ }
38
+
39
+ stateMachine.config.id = input.code;
40
+
41
+ const snapshot = await getStateMachineNextSnapshot({
42
+ machineConfig: stateMachine.config,
43
+ context: input.context,
44
+ currentState: input.currentState,
45
+ event: input.event,
46
+ });
47
+
48
+ response.json({
49
+ state: snapshot.value,
50
+ });
51
+ }
@@ -1,148 +1,148 @@
1
- /**
2
- * Webhooks plugin
3
- */
4
-
5
- import { RpdApplicationConfig, RpdEntityCreateEventPayload, RpdEntityDeleteEventPayload, RpdEntityUpdateEventPayload, RpdServerEventTypes } from "~/types";
6
- import {
7
- RpdServerPluginExtendingAbilities,
8
- RpdServerPluginConfigurableTargetOptions,
9
- RpdConfigurationItemOptions,
10
- IRpdServer,
11
- RapidPlugin,
12
- } from "~/core/server";
13
- import { fetchWithTimeout } from "~/utilities/httpUtility";
14
- import pluginConfig from "./pluginConfig";
15
- import { indexOf } from "lodash";
16
-
17
- export interface Webhook {
18
- name: string;
19
- url: string;
20
- secret: string;
21
- namespace: string;
22
- modelSingularCode: string;
23
- events: string[];
24
- properties: string[];
25
- enabled: boolean;
26
- }
27
-
28
- function listWebhooks(server: IRpdServer) {
29
- const logger = server.getLogger();
30
- logger.info("Loading meta of webhooks...");
31
-
32
- try {
33
- const entityManager = server.getEntityManager("webhook");
34
- return entityManager.findEntities({
35
- filters: [
36
- {
37
- field: "enabled",
38
- operator: "eq",
39
- value: true,
40
- },
41
- ],
42
- });
43
- } catch (error) {
44
- logger.crit("Failed to load meta of webhooks.", { error });
45
- }
46
- }
47
-
48
- class WebhooksPlugin implements RapidPlugin {
49
- #webhooks: Webhook[];
50
-
51
- constructor() {
52
- this.#webhooks = [];
53
- }
54
-
55
- get code(): string {
56
- return "webhooks";
57
- }
58
-
59
- get description(): string {
60
- return null;
61
- }
62
-
63
- get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
64
- return [];
65
- }
66
-
67
- get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
68
- return [];
69
- }
70
-
71
- get configurations(): RpdConfigurationItemOptions[] {
72
- return [];
73
- }
74
-
75
- async registerEventHandlers(server: IRpdServer): Promise<any> {
76
- const events: (keyof RpdServerEventTypes)[] = ["entity.create", "entity.update", "entity.delete"];
77
- for (const event of events) {
78
- server.registerEventHandler(event, this.handleEntityEvent.bind(this, server, event));
79
- }
80
- }
81
-
82
- async configureModels(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
83
- server.appendApplicationConfig({
84
- models: pluginConfig.models,
85
- });
86
- }
87
-
88
- async configureModelProperties(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
89
-
90
- async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
91
-
92
- async onApplicationLoaded(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
93
- this.#webhooks = await listWebhooks(server);
94
- }
95
-
96
- async onApplicationReady(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
97
-
98
- async handleEntityEvent(
99
- server: IRpdServer,
100
- event: keyof RpdServerEventTypes,
101
- sender: RapidPlugin,
102
- payload: RpdEntityCreateEventPayload | RpdEntityUpdateEventPayload | RpdEntityDeleteEventPayload,
103
- ) {
104
- if (sender === this) {
105
- return;
106
- }
107
-
108
- if (payload.namespace === "sys" && payload.modelSingularCode === "webhook") {
109
- this.#webhooks = await listWebhooks(server);
110
- return;
111
- }
112
-
113
- // We will not trigger webhooks if entity changed is in "meta" or "sys" namespaces.
114
- if (payload.namespace === "meta" || payload.namespace === "sys") {
115
- return;
116
- }
117
-
118
- const logger = server.getLogger();
119
-
120
- for (const webhook of this.#webhooks) {
121
- if (indexOf(webhook.events, event) === -1) {
122
- continue;
123
- }
124
-
125
- if (webhook.namespace != payload.namespace || webhook.modelSingularCode !== payload.modelSingularCode) {
126
- continue;
127
- }
128
-
129
- logger.debug(`Triggering webhook. ${webhook.url}`);
130
- // TODO: It's better to trigger webhook through message queue.
131
- const requestBody = { event, payload };
132
- try {
133
- await fetchWithTimeout(webhook.url, {
134
- method: "post",
135
- headers: {
136
- "Content-Type": "application/json",
137
- "x-webhook-secret": webhook.secret || "",
138
- },
139
- body: JSON.stringify(requestBody),
140
- });
141
- } catch (error) {
142
- logger.error("Failed to call webhook.", { error, webhookUrl: webhook.url, requestBody });
143
- }
144
- }
145
- }
146
- }
147
-
148
- export default WebhooksPlugin;
1
+ /**
2
+ * Webhooks plugin
3
+ */
4
+
5
+ import { RpdApplicationConfig, RpdEntityCreateEventPayload, RpdEntityDeleteEventPayload, RpdEntityUpdateEventPayload, RpdServerEventTypes } from "~/types";
6
+ import {
7
+ RpdServerPluginExtendingAbilities,
8
+ RpdServerPluginConfigurableTargetOptions,
9
+ RpdConfigurationItemOptions,
10
+ IRpdServer,
11
+ RapidPlugin,
12
+ } from "~/core/server";
13
+ import { fetchWithTimeout } from "~/utilities/httpUtility";
14
+ import pluginConfig from "./pluginConfig";
15
+ import { indexOf } from "lodash";
16
+
17
+ export interface Webhook {
18
+ name: string;
19
+ url: string;
20
+ secret: string;
21
+ namespace: string;
22
+ modelSingularCode: string;
23
+ events: string[];
24
+ properties: string[];
25
+ enabled: boolean;
26
+ }
27
+
28
+ function listWebhooks(server: IRpdServer) {
29
+ const logger = server.getLogger();
30
+ logger.info("Loading meta of webhooks...");
31
+
32
+ try {
33
+ const entityManager = server.getEntityManager("webhook");
34
+ return entityManager.findEntities({
35
+ filters: [
36
+ {
37
+ field: "enabled",
38
+ operator: "eq",
39
+ value: true,
40
+ },
41
+ ],
42
+ });
43
+ } catch (error) {
44
+ logger.crit("Failed to load meta of webhooks.", { error });
45
+ }
46
+ }
47
+
48
+ class WebhooksPlugin implements RapidPlugin {
49
+ #webhooks: Webhook[];
50
+
51
+ constructor() {
52
+ this.#webhooks = [];
53
+ }
54
+
55
+ get code(): string {
56
+ return "webhooks";
57
+ }
58
+
59
+ get description(): string {
60
+ return null;
61
+ }
62
+
63
+ get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
64
+ return [];
65
+ }
66
+
67
+ get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
68
+ return [];
69
+ }
70
+
71
+ get configurations(): RpdConfigurationItemOptions[] {
72
+ return [];
73
+ }
74
+
75
+ async registerEventHandlers(server: IRpdServer): Promise<any> {
76
+ const events: (keyof RpdServerEventTypes)[] = ["entity.create", "entity.update", "entity.delete"];
77
+ for (const event of events) {
78
+ server.registerEventHandler(event, this.handleEntityEvent.bind(this, server, event));
79
+ }
80
+ }
81
+
82
+ async configureModels(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
83
+ server.appendApplicationConfig({
84
+ models: pluginConfig.models,
85
+ });
86
+ }
87
+
88
+ async configureModelProperties(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
89
+
90
+ async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
91
+
92
+ async onApplicationLoaded(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
93
+ this.#webhooks = await listWebhooks(server);
94
+ }
95
+
96
+ async onApplicationReady(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {}
97
+
98
+ async handleEntityEvent(
99
+ server: IRpdServer,
100
+ event: keyof RpdServerEventTypes,
101
+ sender: RapidPlugin,
102
+ payload: RpdEntityCreateEventPayload | RpdEntityUpdateEventPayload | RpdEntityDeleteEventPayload,
103
+ ) {
104
+ if (sender === this) {
105
+ return;
106
+ }
107
+
108
+ if (payload.namespace === "sys" && payload.modelSingularCode === "webhook") {
109
+ this.#webhooks = await listWebhooks(server);
110
+ return;
111
+ }
112
+
113
+ // We will not trigger webhooks if entity changed is in "meta" or "sys" namespaces.
114
+ if (payload.namespace === "meta" || payload.namespace === "sys") {
115
+ return;
116
+ }
117
+
118
+ const logger = server.getLogger();
119
+
120
+ for (const webhook of this.#webhooks) {
121
+ if (indexOf(webhook.events, event) === -1) {
122
+ continue;
123
+ }
124
+
125
+ if (webhook.namespace != payload.namespace || webhook.modelSingularCode !== payload.modelSingularCode) {
126
+ continue;
127
+ }
128
+
129
+ logger.debug(`Triggering webhook. ${webhook.url}`);
130
+ // TODO: It's better to trigger webhook through message queue.
131
+ const requestBody = { event, payload };
132
+ try {
133
+ await fetchWithTimeout(webhook.url, {
134
+ method: "post",
135
+ headers: {
136
+ "Content-Type": "application/json",
137
+ "x-webhook-secret": webhook.secret || "",
138
+ },
139
+ body: JSON.stringify(requestBody),
140
+ });
141
+ } catch (error) {
142
+ logger.error("Failed to call webhook.", { error, webhookUrl: webhook.url, requestBody });
143
+ }
144
+ }
145
+ }
146
+ }
147
+
148
+ export default WebhooksPlugin;
@@ -1,75 +1,75 @@
1
- import { RpdApplicationConfig } from "~/types";
2
-
3
- export default {
4
- models: [
5
- {
6
- name: "webhook",
7
- namespace: "sys",
8
- singularCode: "webhook",
9
- pluralCode: "webhooks",
10
- schema: "public",
11
- tableName: "sys_webhooks",
12
- properties: [
13
- {
14
- name: "id",
15
- code: "id",
16
- columnName: "id",
17
- type: "integer",
18
- required: true,
19
- autoIncrement: true,
20
- },
21
- {
22
- name: "name",
23
- code: "name",
24
- columnName: "name",
25
- type: "text",
26
- required: true,
27
- },
28
- {
29
- name: "url",
30
- code: "url",
31
- columnName: "url",
32
- type: "text",
33
- required: true,
34
- },
35
- {
36
- name: "secret",
37
- code: "secret",
38
- columnName: "secret",
39
- type: "text",
40
- required: false,
41
- },
42
- {
43
- name: "namespace",
44
- code: "namespace",
45
- columnName: "namespace",
46
- type: "text",
47
- required: true,
48
- },
49
- {
50
- name: "model singular code",
51
- code: "modelSingularCode",
52
- columnName: "model_singular_code",
53
- type: "text",
54
- required: true,
55
- },
56
- {
57
- name: "events",
58
- code: "events",
59
- columnName: "events",
60
- type: "json",
61
- required: false,
62
- },
63
- {
64
- name: "enabled",
65
- code: "enabled",
66
- columnName: "enabled",
67
- type: "boolean",
68
- required: true,
69
- },
70
- ],
71
- },
72
- ],
73
- dataDictionaries: [],
74
- routes: [],
75
- } satisfies RpdApplicationConfig;
1
+ import { RpdApplicationConfig } from "~/types";
2
+
3
+ export default {
4
+ models: [
5
+ {
6
+ name: "webhook",
7
+ namespace: "sys",
8
+ singularCode: "webhook",
9
+ pluralCode: "webhooks",
10
+ schema: "public",
11
+ tableName: "sys_webhooks",
12
+ properties: [
13
+ {
14
+ name: "id",
15
+ code: "id",
16
+ columnName: "id",
17
+ type: "integer",
18
+ required: true,
19
+ autoIncrement: true,
20
+ },
21
+ {
22
+ name: "name",
23
+ code: "name",
24
+ columnName: "name",
25
+ type: "text",
26
+ required: true,
27
+ },
28
+ {
29
+ name: "url",
30
+ code: "url",
31
+ columnName: "url",
32
+ type: "text",
33
+ required: true,
34
+ },
35
+ {
36
+ name: "secret",
37
+ code: "secret",
38
+ columnName: "secret",
39
+ type: "text",
40
+ required: false,
41
+ },
42
+ {
43
+ name: "namespace",
44
+ code: "namespace",
45
+ columnName: "namespace",
46
+ type: "text",
47
+ required: true,
48
+ },
49
+ {
50
+ name: "model singular code",
51
+ code: "modelSingularCode",
52
+ columnName: "model_singular_code",
53
+ type: "text",
54
+ required: true,
55
+ },
56
+ {
57
+ name: "events",
58
+ code: "events",
59
+ columnName: "events",
60
+ type: "json",
61
+ required: false,
62
+ },
63
+ {
64
+ name: "enabled",
65
+ code: "enabled",
66
+ columnName: "enabled",
67
+ type: "boolean",
68
+ required: true,
69
+ },
70
+ ],
71
+ },
72
+ ],
73
+ dataDictionaries: [],
74
+ routes: [],
75
+ } satisfies RpdApplicationConfig;