@ruiapp/rapid-core 0.5.11 → 0.5.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 +7 -7
- package/dist/facilities/cache/CacheFacilityTypes.d.ts +4 -2
- package/dist/facilities/cache/MemoryCache.d.ts +3 -1
- package/dist/index.d.ts +4 -0
- package/dist/index.js +131 -72
- package/dist/utilities/entityUtility.d.ts +1 -0
- package/dist/utilities/passwordUtility.d.ts +14 -0
- package/package.json +2 -2
- package/rollup.config.js +16 -16
- package/src/bootstrapApplicationConfig.ts +638 -638
- package/src/core/actionHandler.ts +22 -22
- package/src/core/eventManager.ts +20 -20
- package/src/core/facility.ts +7 -7
- package/src/core/http/formDataParser.ts +89 -89
- package/src/core/http-types.ts +4 -4
- package/src/core/pluginManager.ts +175 -175
- package/src/core/providers/runtimeProvider.ts +5 -5
- package/src/core/request.ts +95 -95
- package/src/core/response.ts +79 -79
- package/src/core/routeContext.ts +100 -100
- package/src/core/routesBuilder.ts +88 -88
- package/src/core/server.ts +145 -145
- package/src/dataAccess/columnTypeMapper.ts +22 -22
- package/src/dataAccess/dataAccessTypes.ts +163 -163
- package/src/dataAccess/dataAccessor.ts +135 -135
- package/src/dataAccess/entityManager.ts +1910 -1910
- package/src/dataAccess/entityMapper.ts +100 -100
- package/src/dataAccess/propertyMapper.ts +28 -28
- package/src/deno-std/assert/assert.ts +9 -9
- package/src/deno-std/assert/assertion_error.ts +7 -7
- package/src/deno-std/datetime/to_imf.ts +32 -32
- package/src/deno-std/encoding/base64.ts +141 -141
- package/src/deno-std/http/cookie.ts +372 -372
- package/src/facilities/cache/CacheFacilityTypes.ts +29 -27
- package/src/facilities/cache/CacheFactory.ts +31 -31
- package/src/facilities/cache/MemoryCache.ts +58 -42
- package/src/facilities/cache/MemoryCacheProvider.ts +15 -15
- package/src/facilities/log/LogFacility.ts +35 -35
- package/src/helpers/entityHelpers.ts +76 -76
- package/src/helpers/filterHelper.ts +148 -148
- package/src/helpers/inputHelper.ts +11 -11
- package/src/helpers/metaHelper.ts +104 -104
- package/src/helpers/runCollectionEntityActionHandler.ts +57 -57
- package/src/index.ts +67 -63
- package/src/plugins/auth/AuthPlugin.ts +93 -93
- package/src/plugins/auth/actionHandlers/changePassword.ts +60 -61
- package/src/plugins/auth/actionHandlers/createSession.ts +68 -68
- package/src/plugins/auth/actionHandlers/deleteSession.ts +18 -18
- package/src/plugins/auth/actionHandlers/getMyProfile.ts +35 -35
- package/src/plugins/auth/actionHandlers/index.ts +8 -8
- package/src/plugins/auth/actionHandlers/resetPassword.ts +44 -45
- package/src/plugins/auth/models/AccessToken.ts +56 -56
- package/src/plugins/auth/models/index.ts +3 -3
- package/src/plugins/auth/routes/changePassword.ts +15 -15
- package/src/plugins/auth/routes/getMyProfile.ts +15 -15
- package/src/plugins/auth/routes/index.ts +7 -7
- package/src/plugins/auth/routes/resetPassword.ts +15 -15
- package/src/plugins/auth/routes/signin.ts +15 -15
- package/src/plugins/auth/routes/signout.ts +15 -15
- package/src/plugins/auth/services/AuthService.ts +39 -39
- package/src/plugins/cronJob/CronJobPlugin.ts +112 -112
- package/src/plugins/cronJob/CronJobPluginTypes.ts +49 -49
- package/src/plugins/cronJob/actionHandlers/index.ts +4 -4
- package/src/plugins/cronJob/actionHandlers/runCronJob.ts +29 -29
- package/src/plugins/cronJob/routes/index.ts +3 -3
- package/src/plugins/cronJob/routes/runCronJob.ts +15 -15
- package/src/plugins/dataManage/DataManagePlugin.ts +163 -163
- package/src/plugins/dataManage/actionHandlers/addEntityRelations.ts +15 -15
- package/src/plugins/dataManage/actionHandlers/countCollectionEntities.ts +17 -17
- package/src/plugins/dataManage/actionHandlers/createCollectionEntitiesBatch.ts +81 -81
- package/src/plugins/dataManage/actionHandlers/createCollectionEntity.ts +20 -20
- package/src/plugins/dataManage/actionHandlers/deleteCollectionEntities.ts +45 -45
- package/src/plugins/dataManage/actionHandlers/deleteCollectionEntityById.ts +20 -20
- package/src/plugins/dataManage/actionHandlers/findCollectionEntities.ts +27 -27
- package/src/plugins/dataManage/actionHandlers/findCollectionEntityById.ts +30 -30
- package/src/plugins/dataManage/actionHandlers/queryDatabase.ts +22 -22
- package/src/plugins/dataManage/actionHandlers/removeEntityRelations.ts +15 -15
- package/src/plugins/dataManage/actionHandlers/updateCollectionEntityById.ts +38 -38
- package/src/plugins/entityAccessControl/EntityAccessControlPlugin.ts +146 -146
- package/src/plugins/fileManage/FileManagePlugin.ts +52 -52
- package/src/plugins/fileManage/actionHandlers/downloadDocument.ts +65 -65
- package/src/plugins/fileManage/actionHandlers/downloadFile.ts +44 -44
- package/src/plugins/fileManage/actionHandlers/uploadFile.ts +33 -33
- package/src/plugins/fileManage/routes/downloadDocument.ts +15 -15
- package/src/plugins/fileManage/routes/downloadFile.ts +15 -15
- package/src/plugins/fileManage/routes/index.ts +5 -5
- package/src/plugins/fileManage/routes/uploadFile.ts +15 -15
- package/src/plugins/license/LicensePlugin.ts +79 -79
- package/src/plugins/license/LicensePluginTypes.ts +95 -95
- package/src/plugins/license/LicenseService.ts +118 -118
- package/src/plugins/license/actionHandlers/getLicense.ts +18 -18
- package/src/plugins/license/actionHandlers/index.ts +4 -4
- package/src/plugins/license/helpers/certHelper.ts +21 -21
- package/src/plugins/license/helpers/cryptoHelper.ts +47 -47
- package/src/plugins/license/models/index.ts +1 -1
- package/src/plugins/license/routes/getLicense.ts +15 -15
- package/src/plugins/license/routes/index.ts +3 -3
- package/src/plugins/mail/MailPlugin.ts +74 -74
- package/src/plugins/mail/MailPluginTypes.ts +27 -27
- package/src/plugins/mail/MailService.ts +38 -38
- package/src/plugins/mail/actionHandlers/index.ts +3 -3
- package/src/plugins/mail/models/index.ts +1 -1
- package/src/plugins/mail/routes/index.ts +1 -1
- package/src/plugins/metaManage/MetaManagePlugin.ts +530 -530
- package/src/plugins/metaManage/actionHandlers/getMetaModelDetail.ts +10 -10
- package/src/plugins/metaManage/actionHandlers/listMetaModels.ts +9 -9
- package/src/plugins/metaManage/actionHandlers/listMetaRoutes.ts +9 -9
- package/src/plugins/notification/NotificationPlugin.ts +68 -68
- package/src/plugins/notification/NotificationPluginTypes.ts +13 -13
- package/src/plugins/notification/NotificationService.ts +25 -25
- package/src/plugins/notification/actionHandlers/index.ts +3 -3
- package/src/plugins/notification/models/Notification.ts +60 -60
- package/src/plugins/notification/models/index.ts +3 -3
- package/src/plugins/notification/routes/index.ts +1 -1
- package/src/plugins/routeManage/RouteManagePlugin.ts +62 -62
- package/src/plugins/routeManage/actionHandlers/httpProxy.ts +13 -13
- package/src/plugins/sequence/SequencePlugin.ts +146 -146
- package/src/plugins/sequence/SequencePluginTypes.ts +69 -69
- package/src/plugins/sequence/SequenceService.ts +92 -92
- package/src/plugins/sequence/actionHandlers/generateSn.ts +32 -32
- package/src/plugins/sequence/actionHandlers/index.ts +4 -4
- package/src/plugins/sequence/models/SequenceAutoIncrementRecord.ts +49 -49
- package/src/plugins/sequence/models/SequenceRule.ts +42 -42
- package/src/plugins/sequence/models/index.ts +4 -4
- package/src/plugins/sequence/routes/generateSn.ts +15 -15
- package/src/plugins/sequence/routes/index.ts +3 -3
- package/src/plugins/sequence/segment-utility.ts +11 -11
- package/src/plugins/sequence/segments/autoIncrement.ts +90 -90
- package/src/plugins/sequence/segments/dayOfMonth.ts +19 -19
- package/src/plugins/sequence/segments/index.ts +9 -9
- package/src/plugins/sequence/segments/literal.ts +16 -16
- package/src/plugins/sequence/segments/month.ts +19 -19
- package/src/plugins/sequence/segments/parameter.ts +20 -20
- package/src/plugins/sequence/segments/year.ts +19 -19
- package/src/plugins/serverOperation/ServerOperationPlugin.ts +91 -91
- package/src/plugins/serverOperation/ServerOperationPluginTypes.ts +15 -15
- package/src/plugins/serverOperation/actionHandlers/index.ts +4 -4
- package/src/plugins/serverOperation/actionHandlers/runServerOperation.ts +15 -15
- package/src/plugins/setting/SettingPlugin.ts +68 -68
- package/src/plugins/setting/SettingPluginTypes.ts +37 -37
- package/src/plugins/setting/SettingService.ts +213 -213
- package/src/plugins/setting/actionHandlers/getSystemSettingValues.ts +30 -30
- package/src/plugins/setting/actionHandlers/getUserSettingValues.ts +38 -38
- package/src/plugins/setting/actionHandlers/index.ts +6 -6
- package/src/plugins/setting/actionHandlers/setSystemSettingValues.ts +30 -30
- package/src/plugins/setting/models/SystemSettingGroupSetting.ts +57 -57
- package/src/plugins/setting/models/SystemSettingItem.ts +48 -48
- package/src/plugins/setting/models/SystemSettingItemSetting.ts +73 -73
- package/src/plugins/setting/models/UserSettingGroupSetting.ts +57 -57
- package/src/plugins/setting/models/UserSettingItem.ts +55 -55
- package/src/plugins/setting/models/UserSettingItemSetting.ts +73 -73
- package/src/plugins/setting/models/index.ts +8 -8
- package/src/plugins/setting/routes/getSystemSettingValues.ts +15 -15
- package/src/plugins/setting/routes/getUserSettingValues.ts +15 -15
- package/src/plugins/setting/routes/index.ts +5 -5
- package/src/plugins/setting/routes/setSystemSettingValues.ts +15 -15
- package/src/plugins/stateMachine/StateMachinePlugin.ts +196 -196
- package/src/plugins/stateMachine/StateMachinePluginTypes.ts +48 -48
- package/src/plugins/stateMachine/actionHandlers/index.ts +4 -4
- package/src/plugins/stateMachine/actionHandlers/sendStateMachineEvent.ts +54 -54
- package/src/plugins/stateMachine/models/StateMachine.ts +42 -42
- package/src/plugins/stateMachine/models/index.ts +3 -3
- package/src/plugins/stateMachine/routes/index.ts +3 -3
- package/src/plugins/stateMachine/routes/sendStateMachineEvent.ts +15 -15
- package/src/plugins/stateMachine/stateMachineHelper.ts +36 -36
- package/src/plugins/webhooks/WebhooksPlugin.ts +148 -148
- package/src/plugins/webhooks/pluginConfig.ts +75 -75
- package/src/polyfill.ts +5 -5
- package/src/proxy/mod.ts +38 -38
- package/src/proxy/types.ts +21 -21
- package/src/queryBuilder/index.ts +1 -1
- package/src/queryBuilder/queryBuilder.ts +668 -668
- package/src/server.ts +480 -480
- package/src/types.ts +722 -722
- package/src/utilities/accessControlUtility.ts +33 -33
- package/src/utilities/entityUtility.ts +18 -0
- package/src/utilities/errorUtility.ts +15 -15
- package/src/utilities/fsUtility.ts +61 -61
- package/src/utilities/httpUtility.ts +19 -19
- package/src/utilities/jwtUtility.ts +26 -26
- package/src/utilities/passwordUtility.ts +26 -0
- package/src/utilities/pathUtility.ts +14 -14
- package/src/utilities/timeUtility.ts +9 -9
- package/src/utilities/typeUtility.ts +15 -15
- package/tsconfig.json +19 -19
|
@@ -1,146 +1,146 @@
|
|
|
1
|
-
import type { RpdApplicationConfig, RpdDataModelProperty } from "~/types";
|
|
2
|
-
|
|
3
|
-
import {
|
|
4
|
-
IRpdServer,
|
|
5
|
-
RapidPlugin,
|
|
6
|
-
RpdConfigurationItemOptions,
|
|
7
|
-
RpdServerPluginConfigurableTargetOptions,
|
|
8
|
-
RpdServerPluginExtendingAbilities,
|
|
9
|
-
} from "~/core/server";
|
|
10
|
-
import { find, set } from "lodash";
|
|
11
|
-
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
12
|
-
import { isAccessAllowed } from "~/utilities/accessControlUtility";
|
|
13
|
-
import { RouteContext } from "~/core/routeContext";
|
|
14
|
-
|
|
15
|
-
class EntityAccessControlPlugin implements RapidPlugin {
|
|
16
|
-
constructor() {}
|
|
17
|
-
|
|
18
|
-
get code(): string {
|
|
19
|
-
return "entityAccessControl";
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
get description(): string {
|
|
23
|
-
return "";
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
|
|
27
|
-
return [];
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
|
|
31
|
-
return [];
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
get configurations(): RpdConfigurationItemOptions[] {
|
|
35
|
-
return [];
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
async onLoadingApplication(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
39
|
-
const properties: RpdDataModelProperty[] = [
|
|
40
|
-
{
|
|
41
|
-
name: "permissionPolicies",
|
|
42
|
-
code: "permissionPolicies",
|
|
43
|
-
columnName: "permission_policies",
|
|
44
|
-
type: "json",
|
|
45
|
-
},
|
|
46
|
-
];
|
|
47
|
-
server.appendModelProperties("model", properties);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
51
|
-
const logger = server.getLogger();
|
|
52
|
-
logger.info("Configuring entity access checking policies...");
|
|
53
|
-
|
|
54
|
-
const routes = applicationConfig.routes;
|
|
55
|
-
for (const route of routes) {
|
|
56
|
-
const { actions } = route;
|
|
57
|
-
if (!actions) {
|
|
58
|
-
continue;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
for (const action of route.actions) {
|
|
62
|
-
if (action.code === "findCollectionEntityById" || action.code === "findCollectionEntities" || action.code === "countCollectionEntities") {
|
|
63
|
-
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
64
|
-
if (!model) {
|
|
65
|
-
continue;
|
|
66
|
-
}
|
|
67
|
-
const { permissionPolicies } = model;
|
|
68
|
-
if (!permissionPolicies) {
|
|
69
|
-
continue;
|
|
70
|
-
}
|
|
71
|
-
if (permissionPolicies.find) {
|
|
72
|
-
set(action, "config.permissionCheck", permissionPolicies.find);
|
|
73
|
-
}
|
|
74
|
-
} else if (action.code === "createCollectionEntity" || action.code === "createCollectionEntitiesBatch") {
|
|
75
|
-
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
76
|
-
if (!model) {
|
|
77
|
-
continue;
|
|
78
|
-
}
|
|
79
|
-
const { permissionPolicies } = model;
|
|
80
|
-
if (!permissionPolicies) {
|
|
81
|
-
continue;
|
|
82
|
-
}
|
|
83
|
-
if (permissionPolicies.create) {
|
|
84
|
-
set(action, "config.permissionCheck", permissionPolicies.create);
|
|
85
|
-
}
|
|
86
|
-
} else if (action.code === "updateCollectionEntityById" || action.code === "addEntityRelations" || action.code === "removeEntityRelations") {
|
|
87
|
-
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
88
|
-
if (!model) {
|
|
89
|
-
continue;
|
|
90
|
-
}
|
|
91
|
-
const { permissionPolicies } = model;
|
|
92
|
-
if (!permissionPolicies) {
|
|
93
|
-
continue;
|
|
94
|
-
}
|
|
95
|
-
if (permissionPolicies.update) {
|
|
96
|
-
set(action, "config.permissionCheck", permissionPolicies.update);
|
|
97
|
-
}
|
|
98
|
-
} else if (action.code === "deleteCollectionEntityById" || action.code === "deleteCollectionEntities") {
|
|
99
|
-
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
100
|
-
if (!model) {
|
|
101
|
-
continue;
|
|
102
|
-
}
|
|
103
|
-
const { permissionPolicies } = model;
|
|
104
|
-
if (!permissionPolicies) {
|
|
105
|
-
continue;
|
|
106
|
-
}
|
|
107
|
-
if (permissionPolicies.delete) {
|
|
108
|
-
set(action, "config.permissionCheck", permissionPolicies.delete);
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
async onPrepareRouteContext(server: IRpdServer, routeContext: RouteContext) {
|
|
116
|
-
const userId = routeContext.state.userId;
|
|
117
|
-
if (!userId) {
|
|
118
|
-
return;
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
const actions = await server.queryDatabaseObject(
|
|
122
|
-
`select distinct a.* from sys_actions a
|
|
123
|
-
inner join oc_role_sys_action_links ra on a.id = ra.action_id
|
|
124
|
-
inner join oc_role_user_links ru on ru.role_id = ra.role_id
|
|
125
|
-
where ru.user_id = $1;`,
|
|
126
|
-
[userId],
|
|
127
|
-
);
|
|
128
|
-
routeContext.state.allowedActions = actions.map((item) => item.code);
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
async beforeRunRouteActions(server: IRpdServer, handlerContext: ActionHandlerContext): Promise<any> {
|
|
132
|
-
// Check permission
|
|
133
|
-
const { routerContext } = handlerContext;
|
|
134
|
-
const { routeConfig } = routerContext;
|
|
135
|
-
for (const actionConfig of routeConfig.actions) {
|
|
136
|
-
const permissionCheck = actionConfig.config?.permissionCheck;
|
|
137
|
-
if (permissionCheck) {
|
|
138
|
-
if (!isAccessAllowed(permissionCheck, routerContext.state.allowedActions || [])) {
|
|
139
|
-
throw new Error(`Your action of '${actionConfig.code}' is not permitted.`);
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
}
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
export default EntityAccessControlPlugin;
|
|
1
|
+
import type { RpdApplicationConfig, RpdDataModelProperty } from "~/types";
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
IRpdServer,
|
|
5
|
+
RapidPlugin,
|
|
6
|
+
RpdConfigurationItemOptions,
|
|
7
|
+
RpdServerPluginConfigurableTargetOptions,
|
|
8
|
+
RpdServerPluginExtendingAbilities,
|
|
9
|
+
} from "~/core/server";
|
|
10
|
+
import { find, set } from "lodash";
|
|
11
|
+
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
12
|
+
import { isAccessAllowed } from "~/utilities/accessControlUtility";
|
|
13
|
+
import { RouteContext } from "~/core/routeContext";
|
|
14
|
+
|
|
15
|
+
class EntityAccessControlPlugin implements RapidPlugin {
|
|
16
|
+
constructor() {}
|
|
17
|
+
|
|
18
|
+
get code(): string {
|
|
19
|
+
return "entityAccessControl";
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
get description(): string {
|
|
23
|
+
return "";
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
|
|
27
|
+
return [];
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
|
|
31
|
+
return [];
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
get configurations(): RpdConfigurationItemOptions[] {
|
|
35
|
+
return [];
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
async onLoadingApplication(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
39
|
+
const properties: RpdDataModelProperty[] = [
|
|
40
|
+
{
|
|
41
|
+
name: "permissionPolicies",
|
|
42
|
+
code: "permissionPolicies",
|
|
43
|
+
columnName: "permission_policies",
|
|
44
|
+
type: "json",
|
|
45
|
+
},
|
|
46
|
+
];
|
|
47
|
+
server.appendModelProperties("model", properties);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
51
|
+
const logger = server.getLogger();
|
|
52
|
+
logger.info("Configuring entity access checking policies...");
|
|
53
|
+
|
|
54
|
+
const routes = applicationConfig.routes;
|
|
55
|
+
for (const route of routes) {
|
|
56
|
+
const { actions } = route;
|
|
57
|
+
if (!actions) {
|
|
58
|
+
continue;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
for (const action of route.actions) {
|
|
62
|
+
if (action.code === "findCollectionEntityById" || action.code === "findCollectionEntities" || action.code === "countCollectionEntities") {
|
|
63
|
+
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
64
|
+
if (!model) {
|
|
65
|
+
continue;
|
|
66
|
+
}
|
|
67
|
+
const { permissionPolicies } = model;
|
|
68
|
+
if (!permissionPolicies) {
|
|
69
|
+
continue;
|
|
70
|
+
}
|
|
71
|
+
if (permissionPolicies.find) {
|
|
72
|
+
set(action, "config.permissionCheck", permissionPolicies.find);
|
|
73
|
+
}
|
|
74
|
+
} else if (action.code === "createCollectionEntity" || action.code === "createCollectionEntitiesBatch") {
|
|
75
|
+
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
76
|
+
if (!model) {
|
|
77
|
+
continue;
|
|
78
|
+
}
|
|
79
|
+
const { permissionPolicies } = model;
|
|
80
|
+
if (!permissionPolicies) {
|
|
81
|
+
continue;
|
|
82
|
+
}
|
|
83
|
+
if (permissionPolicies.create) {
|
|
84
|
+
set(action, "config.permissionCheck", permissionPolicies.create);
|
|
85
|
+
}
|
|
86
|
+
} else if (action.code === "updateCollectionEntityById" || action.code === "addEntityRelations" || action.code === "removeEntityRelations") {
|
|
87
|
+
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
88
|
+
if (!model) {
|
|
89
|
+
continue;
|
|
90
|
+
}
|
|
91
|
+
const { permissionPolicies } = model;
|
|
92
|
+
if (!permissionPolicies) {
|
|
93
|
+
continue;
|
|
94
|
+
}
|
|
95
|
+
if (permissionPolicies.update) {
|
|
96
|
+
set(action, "config.permissionCheck", permissionPolicies.update);
|
|
97
|
+
}
|
|
98
|
+
} else if (action.code === "deleteCollectionEntityById" || action.code === "deleteCollectionEntities") {
|
|
99
|
+
const model = find(applicationConfig.models, (item) => item.singularCode === action.config.singularCode);
|
|
100
|
+
if (!model) {
|
|
101
|
+
continue;
|
|
102
|
+
}
|
|
103
|
+
const { permissionPolicies } = model;
|
|
104
|
+
if (!permissionPolicies) {
|
|
105
|
+
continue;
|
|
106
|
+
}
|
|
107
|
+
if (permissionPolicies.delete) {
|
|
108
|
+
set(action, "config.permissionCheck", permissionPolicies.delete);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async onPrepareRouteContext(server: IRpdServer, routeContext: RouteContext) {
|
|
116
|
+
const userId = routeContext.state.userId;
|
|
117
|
+
if (!userId) {
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const actions = await server.queryDatabaseObject(
|
|
122
|
+
`select distinct a.* from sys_actions a
|
|
123
|
+
inner join oc_role_sys_action_links ra on a.id = ra.action_id
|
|
124
|
+
inner join oc_role_user_links ru on ru.role_id = ra.role_id
|
|
125
|
+
where ru.user_id = $1;`,
|
|
126
|
+
[userId],
|
|
127
|
+
);
|
|
128
|
+
routeContext.state.allowedActions = actions.map((item) => item.code);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
async beforeRunRouteActions(server: IRpdServer, handlerContext: ActionHandlerContext): Promise<any> {
|
|
132
|
+
// Check permission
|
|
133
|
+
const { routerContext } = handlerContext;
|
|
134
|
+
const { routeConfig } = routerContext;
|
|
135
|
+
for (const actionConfig of routeConfig.actions) {
|
|
136
|
+
const permissionCheck = actionConfig.config?.permissionCheck;
|
|
137
|
+
if (permissionCheck) {
|
|
138
|
+
if (!isAccessAllowed(permissionCheck, routerContext.state.allowedActions || [])) {
|
|
139
|
+
throw new Error(`Your action of '${actionConfig.code}' is not permitted.`);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
export default EntityAccessControlPlugin;
|
|
@@ -1,52 +1,52 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* File manager plugin
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
import { RpdApplicationConfig } from "~/types";
|
|
6
|
-
|
|
7
|
-
import * as downloadDocumentActionHandler from "./actionHandlers/downloadDocument";
|
|
8
|
-
import * as downloadFileActionHandler from "./actionHandlers/downloadFile";
|
|
9
|
-
import * as uploadFileActionHandler from "./actionHandlers/uploadFile";
|
|
10
|
-
import {
|
|
11
|
-
IRpdServer,
|
|
12
|
-
RapidPlugin,
|
|
13
|
-
RpdConfigurationItemOptions,
|
|
14
|
-
RpdServerPluginConfigurableTargetOptions,
|
|
15
|
-
RpdServerPluginExtendingAbilities,
|
|
16
|
-
} from "~/core/server";
|
|
17
|
-
|
|
18
|
-
import pluginRoutes from "./routes";
|
|
19
|
-
|
|
20
|
-
class FileManager implements RapidPlugin {
|
|
21
|
-
get code(): string {
|
|
22
|
-
return "fileManager";
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
get description(): string {
|
|
26
|
-
return null;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
|
|
30
|
-
return [];
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
|
|
34
|
-
return [];
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
get configurations(): RpdConfigurationItemOptions[] {
|
|
38
|
-
return [];
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
async registerActionHandlers(server: IRpdServer): Promise<any> {
|
|
42
|
-
server.registerActionHandler(this, downloadDocumentActionHandler);
|
|
43
|
-
server.registerActionHandler(this, downloadFileActionHandler);
|
|
44
|
-
server.registerActionHandler(this, uploadFileActionHandler);
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
48
|
-
server.appendApplicationConfig({ routes: pluginRoutes });
|
|
49
|
-
}
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
export default FileManager;
|
|
1
|
+
/**
|
|
2
|
+
* File manager plugin
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { RpdApplicationConfig } from "~/types";
|
|
6
|
+
|
|
7
|
+
import * as downloadDocumentActionHandler from "./actionHandlers/downloadDocument";
|
|
8
|
+
import * as downloadFileActionHandler from "./actionHandlers/downloadFile";
|
|
9
|
+
import * as uploadFileActionHandler from "./actionHandlers/uploadFile";
|
|
10
|
+
import {
|
|
11
|
+
IRpdServer,
|
|
12
|
+
RapidPlugin,
|
|
13
|
+
RpdConfigurationItemOptions,
|
|
14
|
+
RpdServerPluginConfigurableTargetOptions,
|
|
15
|
+
RpdServerPluginExtendingAbilities,
|
|
16
|
+
} from "~/core/server";
|
|
17
|
+
|
|
18
|
+
import pluginRoutes from "./routes";
|
|
19
|
+
|
|
20
|
+
class FileManager implements RapidPlugin {
|
|
21
|
+
get code(): string {
|
|
22
|
+
return "fileManager";
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
get description(): string {
|
|
26
|
+
return null;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
get extendingAbilities(): RpdServerPluginExtendingAbilities[] {
|
|
30
|
+
return [];
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
get configurableTargets(): RpdServerPluginConfigurableTargetOptions[] {
|
|
34
|
+
return [];
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
get configurations(): RpdConfigurationItemOptions[] {
|
|
38
|
+
return [];
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
async registerActionHandlers(server: IRpdServer): Promise<any> {
|
|
42
|
+
server.registerActionHandler(this, downloadDocumentActionHandler);
|
|
43
|
+
server.registerActionHandler(this, downloadFileActionHandler);
|
|
44
|
+
server.registerActionHandler(this, uploadFileActionHandler);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
async configureRoutes(server: IRpdServer, applicationConfig: RpdApplicationConfig): Promise<any> {
|
|
48
|
+
server.appendApplicationConfig({ routes: pluginRoutes });
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export default FileManager;
|
|
@@ -1,65 +1,65 @@
|
|
|
1
|
-
import path from "path";
|
|
2
|
-
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
3
|
-
import { RapidPlugin } from "~/core/server";
|
|
4
|
-
import { readFile } from "~/utilities/fsUtility";
|
|
5
|
-
import { getFileBaseName } from "~/utilities/pathUtility";
|
|
6
|
-
|
|
7
|
-
export const code = "downloadDocument";
|
|
8
|
-
|
|
9
|
-
export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: any) {
|
|
10
|
-
const { server, applicationConfig, routerContext: routeContext, input } = ctx;
|
|
11
|
-
const { request, response } = routeContext;
|
|
12
|
-
|
|
13
|
-
const documentDataAccessor = ctx.server.getDataAccessor({
|
|
14
|
-
singularCode: "ecm_document",
|
|
15
|
-
});
|
|
16
|
-
const revisionDataAccessor = ctx.server.getDataAccessor({
|
|
17
|
-
singularCode: "ecm_revision",
|
|
18
|
-
});
|
|
19
|
-
const storageDataAccessor = ctx.server.getDataAccessor({
|
|
20
|
-
singularCode: "ecm_storage_object",
|
|
21
|
-
});
|
|
22
|
-
|
|
23
|
-
let storageObjectId = 0;
|
|
24
|
-
let fileName: string;
|
|
25
|
-
let { revisionId, documentId } = input;
|
|
26
|
-
if (revisionId) {
|
|
27
|
-
const revision = await revisionDataAccessor.findById(revisionId, routeContext?.getDbTransactionClient());
|
|
28
|
-
if (!revision) {
|
|
29
|
-
ctx.output = { error: new Error(`Revision with id "${revisionId}" was not found.`) };
|
|
30
|
-
return;
|
|
31
|
-
}
|
|
32
|
-
storageObjectId = revision.storage_object_id;
|
|
33
|
-
|
|
34
|
-
documentId = revision.document_id;
|
|
35
|
-
const document = await documentDataAccessor.findById(documentId, routeContext?.getDbTransactionClient());
|
|
36
|
-
if (!document) {
|
|
37
|
-
ctx.output = { error: new Error(`Document with id "${documentId}" was not found.`) };
|
|
38
|
-
return;
|
|
39
|
-
}
|
|
40
|
-
fileName = `${getFileBaseName(document.name!)}${revision.ext_name}`;
|
|
41
|
-
} else if (documentId) {
|
|
42
|
-
const document = await documentDataAccessor.findById(documentId, routeContext?.getDbTransactionClient());
|
|
43
|
-
if (!document) {
|
|
44
|
-
ctx.output = { error: new Error(`Document with id "${documentId}" was not found.`) };
|
|
45
|
-
return;
|
|
46
|
-
}
|
|
47
|
-
storageObjectId = document.storage_object_id;
|
|
48
|
-
fileName = document.name;
|
|
49
|
-
} else {
|
|
50
|
-
ctx.output = { error: new Error(`Parameter "revisionId" or "documentId" must be provided.`) };
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
const storageObject = await storageDataAccessor.findById(storageObjectId, routeContext?.getDbTransactionClient());
|
|
55
|
-
if (!storageObject) {
|
|
56
|
-
ctx.output = { error: new Error(`Storage object with id "${storageObjectId}" was not found.`) };
|
|
57
|
-
return;
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
const fileKey = storageObject.key;
|
|
61
|
-
const filePathName = path.join(server.config.localFileStoragePath, fileKey);
|
|
62
|
-
|
|
63
|
-
response.body = await readFile(filePathName);
|
|
64
|
-
response.headers.set("Content-Disposition", `attachment; filename="${encodeURIComponent(fileName)}"`);
|
|
65
|
-
}
|
|
1
|
+
import path from "path";
|
|
2
|
+
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
3
|
+
import { RapidPlugin } from "~/core/server";
|
|
4
|
+
import { readFile } from "~/utilities/fsUtility";
|
|
5
|
+
import { getFileBaseName } from "~/utilities/pathUtility";
|
|
6
|
+
|
|
7
|
+
export const code = "downloadDocument";
|
|
8
|
+
|
|
9
|
+
export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: any) {
|
|
10
|
+
const { server, applicationConfig, routerContext: routeContext, input } = ctx;
|
|
11
|
+
const { request, response } = routeContext;
|
|
12
|
+
|
|
13
|
+
const documentDataAccessor = ctx.server.getDataAccessor({
|
|
14
|
+
singularCode: "ecm_document",
|
|
15
|
+
});
|
|
16
|
+
const revisionDataAccessor = ctx.server.getDataAccessor({
|
|
17
|
+
singularCode: "ecm_revision",
|
|
18
|
+
});
|
|
19
|
+
const storageDataAccessor = ctx.server.getDataAccessor({
|
|
20
|
+
singularCode: "ecm_storage_object",
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
let storageObjectId = 0;
|
|
24
|
+
let fileName: string;
|
|
25
|
+
let { revisionId, documentId } = input;
|
|
26
|
+
if (revisionId) {
|
|
27
|
+
const revision = await revisionDataAccessor.findById(revisionId, routeContext?.getDbTransactionClient());
|
|
28
|
+
if (!revision) {
|
|
29
|
+
ctx.output = { error: new Error(`Revision with id "${revisionId}" was not found.`) };
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
storageObjectId = revision.storage_object_id;
|
|
33
|
+
|
|
34
|
+
documentId = revision.document_id;
|
|
35
|
+
const document = await documentDataAccessor.findById(documentId, routeContext?.getDbTransactionClient());
|
|
36
|
+
if (!document) {
|
|
37
|
+
ctx.output = { error: new Error(`Document with id "${documentId}" was not found.`) };
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
fileName = `${getFileBaseName(document.name!)}${revision.ext_name}`;
|
|
41
|
+
} else if (documentId) {
|
|
42
|
+
const document = await documentDataAccessor.findById(documentId, routeContext?.getDbTransactionClient());
|
|
43
|
+
if (!document) {
|
|
44
|
+
ctx.output = { error: new Error(`Document with id "${documentId}" was not found.`) };
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
storageObjectId = document.storage_object_id;
|
|
48
|
+
fileName = document.name;
|
|
49
|
+
} else {
|
|
50
|
+
ctx.output = { error: new Error(`Parameter "revisionId" or "documentId" must be provided.`) };
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const storageObject = await storageDataAccessor.findById(storageObjectId, routeContext?.getDbTransactionClient());
|
|
55
|
+
if (!storageObject) {
|
|
56
|
+
ctx.output = { error: new Error(`Storage object with id "${storageObjectId}" was not found.`) };
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const fileKey = storageObject.key;
|
|
61
|
+
const filePathName = path.join(server.config.localFileStoragePath, fileKey);
|
|
62
|
+
|
|
63
|
+
response.body = await readFile(filePathName);
|
|
64
|
+
response.headers.set("Content-Disposition", `attachment; filename="${encodeURIComponent(fileName)}"`);
|
|
65
|
+
}
|
|
@@ -1,44 +1,44 @@
|
|
|
1
|
-
import path from "path";
|
|
2
|
-
import { readFile } from "~/utilities/fsUtility";
|
|
3
|
-
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
4
|
-
import { RapidPlugin } from "~/core/server";
|
|
5
|
-
|
|
6
|
-
export type DownloadFileInput = {
|
|
7
|
-
fileId?: string;
|
|
8
|
-
fileKey?: string;
|
|
9
|
-
fileName?: string;
|
|
10
|
-
inline?: boolean;
|
|
11
|
-
};
|
|
12
|
-
|
|
13
|
-
export const code = "downloadFile";
|
|
14
|
-
|
|
15
|
-
export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: any) {
|
|
16
|
-
const { server, applicationConfig, routerContext: routeContext } = ctx;
|
|
17
|
-
const { request, response } = routeContext;
|
|
18
|
-
//TODO: only public files can download by this handler
|
|
19
|
-
|
|
20
|
-
const input: DownloadFileInput = ctx.input;
|
|
21
|
-
|
|
22
|
-
let fileKey: string = input.fileKey;
|
|
23
|
-
|
|
24
|
-
if (!fileKey && input.fileId) {
|
|
25
|
-
const dataAccessor = ctx.server.getDataAccessor({
|
|
26
|
-
singularCode: "ecm_storage_object",
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
const storageObject = await dataAccessor.findById(input.fileId, routeContext?.getDbTransactionClient());
|
|
30
|
-
if (!storageObject) {
|
|
31
|
-
ctx.output = { error: new Error("Storage object not found.") };
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
fileKey = storageObject.key;
|
|
36
|
-
}
|
|
37
|
-
const filePathName = path.join(server.config.localFileStoragePath, fileKey);
|
|
38
|
-
const attachmentFileName = input.fileName || path.basename(fileKey);
|
|
39
|
-
|
|
40
|
-
response.body = await readFile(filePathName);
|
|
41
|
-
|
|
42
|
-
const dispositionType = input.inline ? "inline" : "attachment";
|
|
43
|
-
response.headers.set("Content-Disposition", `${dispositionType}; filename="${encodeURIComponent(attachmentFileName)}"`);
|
|
44
|
-
}
|
|
1
|
+
import path from "path";
|
|
2
|
+
import { readFile } from "~/utilities/fsUtility";
|
|
3
|
+
import { ActionHandlerContext } from "~/core/actionHandler";
|
|
4
|
+
import { RapidPlugin } from "~/core/server";
|
|
5
|
+
|
|
6
|
+
export type DownloadFileInput = {
|
|
7
|
+
fileId?: string;
|
|
8
|
+
fileKey?: string;
|
|
9
|
+
fileName?: string;
|
|
10
|
+
inline?: boolean;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
export const code = "downloadFile";
|
|
14
|
+
|
|
15
|
+
export async function handler(plugin: RapidPlugin, ctx: ActionHandlerContext, options: any) {
|
|
16
|
+
const { server, applicationConfig, routerContext: routeContext } = ctx;
|
|
17
|
+
const { request, response } = routeContext;
|
|
18
|
+
//TODO: only public files can download by this handler
|
|
19
|
+
|
|
20
|
+
const input: DownloadFileInput = ctx.input;
|
|
21
|
+
|
|
22
|
+
let fileKey: string = input.fileKey;
|
|
23
|
+
|
|
24
|
+
if (!fileKey && input.fileId) {
|
|
25
|
+
const dataAccessor = ctx.server.getDataAccessor({
|
|
26
|
+
singularCode: "ecm_storage_object",
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
const storageObject = await dataAccessor.findById(input.fileId, routeContext?.getDbTransactionClient());
|
|
30
|
+
if (!storageObject) {
|
|
31
|
+
ctx.output = { error: new Error("Storage object not found.") };
|
|
32
|
+
return;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
fileKey = storageObject.key;
|
|
36
|
+
}
|
|
37
|
+
const filePathName = path.join(server.config.localFileStoragePath, fileKey);
|
|
38
|
+
const attachmentFileName = input.fileName || path.basename(fileKey);
|
|
39
|
+
|
|
40
|
+
response.body = await readFile(filePathName);
|
|
41
|
+
|
|
42
|
+
const dispositionType = input.inline ? "inline" : "attachment";
|
|
43
|
+
response.headers.set("Content-Disposition", `${dispositionType}; filename="${encodeURIComponent(attachmentFileName)}"`);
|
|
44
|
+
}
|