@sd-angular/core 19.0.0-beta.71 → 19.0.0-beta.73
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/components/chart/index.d.ts +4 -0
- package/components/chart/src/bar-chart.component.d.ts +19 -0
- package/components/chart/src/doughnut-chart.component.d.ts +16 -0
- package/components/chart/src/line-chart.component.d.ts +16 -0
- package/components/chart/src/pie-chart.component.d.ts +16 -0
- package/components/index.d.ts +1 -0
- package/components/modal/src/modal.component.d.ts +1 -1
- package/components/section/src/section.component.d.ts +3 -3
- package/components/upload-file/src/configurations/upload-file.configuration.d.ts +33 -0
- package/components/upload-file/src/upload-file.component.d.ts +5 -2
- package/components/workflow/src/models/form-generic-expression.model.d.ts +1 -0
- package/components/workflow/src/pipes/html.pipe.d.ts +4 -4
- package/fesm2022/sd-angular-core-components-chart.mjs +284 -0
- package/fesm2022/sd-angular-core-components-chart.mjs.map +1 -0
- package/fesm2022/sd-angular-core-components-section.mjs +5 -5
- package/fesm2022/sd-angular-core-components-section.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-side-drawer.mjs +2 -2
- package/fesm2022/sd-angular-core-components-side-drawer.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-table.mjs +7 -5
- package/fesm2022/sd-angular-core-components-table.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-upload-file.mjs +61 -9
- package/fesm2022/sd-angular-core-components-upload-file.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components-workflow.mjs +125 -118
- package/fesm2022/sd-angular-core-components-workflow.mjs.map +1 -1
- package/fesm2022/sd-angular-core-components.mjs +1 -0
- package/fesm2022/sd-angular-core-components.mjs.map +1 -1
- package/fesm2022/sd-angular-core-modules-layout.mjs +2 -3
- package/fesm2022/sd-angular-core-modules-layout.mjs.map +1 -1
- package/fesm2022/sd-angular-core-modules-permission.mjs +160 -74
- package/fesm2022/sd-angular-core-modules-permission.mjs.map +1 -1
- package/fesm2022/sd-angular-core-utilities-extensions.mjs +27 -35
- package/fesm2022/sd-angular-core-utilities-extensions.mjs.map +1 -1
- package/fesm2022/sd-angular-core.mjs +0 -1
- package/fesm2022/sd-angular-core.mjs.map +1 -1
- package/modules/layout/services/menu/menu.model.d.ts +1 -0
- package/modules/permission/src/configurations/permission.configuration.d.ts +56 -2
- package/modules/permission/src/directives/permission.directive.d.ts +5 -8
- package/modules/permission/src/guards/permission.guard.d.ts +2 -1
- package/modules/permission/src/services/permission.service.d.ts +6 -9
- package/package.json +88 -91
- package/public-api.d.ts +0 -1
- package/sd-angular-core-19.0.0-beta.73.tgz +0 -0
- package/fesm2022/sd-angular-core-guards-permission.mjs +0 -155
- package/fesm2022/sd-angular-core-guards-permission.mjs.map +0 -1
- package/fesm2022/sd-angular-core-guards.mjs +0 -6
- package/fesm2022/sd-angular-core-guards.mjs.map +0 -1
- package/guards/index.d.ts +0 -1
- package/guards/permission/index.d.ts +0 -4
- package/guards/permission/src/configurations/index.d.ts +0 -1
- package/guards/permission/src/configurations/permission.configuration.d.ts +0 -8
- package/guards/permission/src/directives/index.d.ts +0 -1
- package/guards/permission/src/directives/permission.directive.d.ts +0 -12
- package/guards/permission/src/guards/index.d.ts +0 -1
- package/guards/permission/src/guards/permission.guard.d.ts +0 -13
- package/guards/permission/src/services/index.d.ts +0 -1
- package/guards/permission/src/services/permission.service.d.ts +0 -15
- package/sd-angular-core-19.0.0-beta.71.tgz +0 -0
|
@@ -1,67 +1,144 @@
|
|
|
1
1
|
import * as i0 from '@angular/core';
|
|
2
|
-
import { InjectionToken,
|
|
2
|
+
import { InjectionToken, inject, Injectable, TemplateRef, ViewContainerRef, input, effect, Directive, Inject } from '@angular/core';
|
|
3
|
+
import { SdCacheService } from '@sd-angular/core/services/cache';
|
|
3
4
|
import { ArrayUtilities } from '@sd-angular/core/utilities/extensions';
|
|
4
5
|
import { SdResolveMaybeAsync } from '@sd-angular/core/utilities';
|
|
5
|
-
import * as i1 from '@sd-angular/core/services/cache';
|
|
6
6
|
|
|
7
|
+
/**
|
|
8
|
+
* InjectionToken cho cấu hình permission.
|
|
9
|
+
*
|
|
10
|
+
* Ví dụ provider:
|
|
11
|
+
* {
|
|
12
|
+
* provide: SD_PERMISSION_CONFIGURATION,
|
|
13
|
+
* useValue: {
|
|
14
|
+
* disabled: false,
|
|
15
|
+
* loadPermissions: () => ['SAMPLE_C_EMPLOYEE_VIEW'],
|
|
16
|
+
* onForbiden: () => router.navigateByUrl('/layout/forbidden')
|
|
17
|
+
* }
|
|
18
|
+
* }
|
|
19
|
+
*/
|
|
7
20
|
const SD_PERMISSION_CONFIGURATION = new InjectionToken('sd-permission.configuration');
|
|
8
21
|
|
|
9
22
|
class SdPermissionService {
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
#
|
|
13
|
-
#
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
default: [],
|
|
21
|
-
});
|
|
23
|
+
#permissionMapByKey = {};
|
|
24
|
+
#configuration = inject(SD_PERMISSION_CONFIGURATION);
|
|
25
|
+
#cacheService = inject(SdCacheService);
|
|
26
|
+
#permissionsByKey = this.#cacheService.create('212a51fa-38d5-43b2-bd46-922d85950ba3', {
|
|
27
|
+
type: 'session',
|
|
28
|
+
default: {},
|
|
29
|
+
});
|
|
30
|
+
#loadedKeys = new Set();
|
|
31
|
+
constructor() {
|
|
32
|
+
this.#validateDuplicateConfigKeys();
|
|
22
33
|
}
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
34
|
+
#getConfigurations = () => {
|
|
35
|
+
const config = this.#configuration;
|
|
36
|
+
if (!config) {
|
|
37
|
+
return [];
|
|
38
|
+
}
|
|
39
|
+
return Array.isArray(config) ? config : [config];
|
|
40
|
+
};
|
|
41
|
+
#normalizeKey = (key) => {
|
|
42
|
+
return key === undefined ? '__undefined__' : key;
|
|
43
|
+
};
|
|
44
|
+
#validateDuplicateConfigKeys = () => {
|
|
45
|
+
const seen = new Set();
|
|
46
|
+
for (const config of this.#getConfigurations()) {
|
|
47
|
+
const normalizedKey = this.#normalizeKey(config.key);
|
|
48
|
+
if (seen.has(normalizedKey)) {
|
|
49
|
+
const keyLabel = config.key === undefined ? 'undefined' : config.key;
|
|
50
|
+
throw new Error(`[Permission] Duplicate permission configuration key: ${keyLabel}`);
|
|
40
51
|
}
|
|
52
|
+
seen.add(normalizedKey);
|
|
41
53
|
}
|
|
42
54
|
};
|
|
43
|
-
|
|
55
|
+
#getConfigurationByKey = (key) => {
|
|
56
|
+
return this.#getConfigurations().find(config => config.key === key);
|
|
57
|
+
};
|
|
58
|
+
#getEffectivePermissionKey = (key) => {
|
|
59
|
+
if (this.#getConfigurationByKey(key)) {
|
|
60
|
+
return key;
|
|
61
|
+
}
|
|
62
|
+
// Portal-level config uses key = undefined and acts as default fallback
|
|
63
|
+
if (key !== undefined && this.#getConfigurationByKey(undefined)) {
|
|
64
|
+
return undefined;
|
|
65
|
+
}
|
|
66
|
+
return key;
|
|
67
|
+
};
|
|
68
|
+
#setPermissionsForKey = (normalizedKey, permissions) => {
|
|
69
|
+
const distinctPermissions = ArrayUtilities.distinct(permissions || []);
|
|
70
|
+
const current = this.#permissionsByKey.get() || {};
|
|
71
|
+
this.#permissionsByKey.set({
|
|
72
|
+
...current,
|
|
73
|
+
[normalizedKey]: distinctPermissions,
|
|
74
|
+
});
|
|
75
|
+
const permissionMap = {};
|
|
76
|
+
distinctPermissions.forEach(permission => {
|
|
77
|
+
permissionMap[permission] = true;
|
|
78
|
+
});
|
|
79
|
+
this.#permissionMapByKey[normalizedKey] = permissionMap;
|
|
80
|
+
};
|
|
81
|
+
loadPermissions = async (key) => {
|
|
82
|
+
const effectiveKey = this.#getEffectivePermissionKey(key);
|
|
83
|
+
const normalizedKey = this.#normalizeKey(effectiveKey);
|
|
84
|
+
if (this.#loadedKeys.has(normalizedKey)) {
|
|
85
|
+
return this.#permissionsByKey.get()?.[normalizedKey] || [];
|
|
86
|
+
}
|
|
87
|
+
const configuration = this.#getConfigurationByKey(effectiveKey);
|
|
88
|
+
if (!configuration) {
|
|
89
|
+
this.#setPermissionsForKey(normalizedKey, []);
|
|
90
|
+
this.#loadedKeys.add(normalizedKey);
|
|
91
|
+
return [];
|
|
92
|
+
}
|
|
93
|
+
try {
|
|
94
|
+
const permissions = await SdResolveMaybeAsync(configuration.loadPermissions());
|
|
95
|
+
this.#setPermissionsForKey(normalizedKey, permissions || []);
|
|
96
|
+
}
|
|
97
|
+
catch (err) {
|
|
98
|
+
console.error(err);
|
|
99
|
+
this.#setPermissionsForKey(normalizedKey, []);
|
|
100
|
+
}
|
|
101
|
+
finally {
|
|
102
|
+
this.#loadedKeys.add(normalizedKey);
|
|
103
|
+
}
|
|
104
|
+
return this.#permissionsByKey.get()?.[normalizedKey] || [];
|
|
105
|
+
};
|
|
106
|
+
loadAllPermissions = async () => {
|
|
107
|
+
const configurations = this.#getConfigurations();
|
|
108
|
+
if (!configurations.length) {
|
|
109
|
+
await this.loadPermissions(undefined);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
await Promise.all(configurations.map(config => this.loadPermissions(config.key)));
|
|
113
|
+
};
|
|
114
|
+
hasPermission = (permission, key) => {
|
|
44
115
|
if (!permission?.toString()) {
|
|
45
116
|
return true;
|
|
46
117
|
}
|
|
47
|
-
|
|
118
|
+
const effectiveKey = this.#getEffectivePermissionKey(key);
|
|
119
|
+
const configuration = this.#getConfigurationByKey(effectiveKey);
|
|
120
|
+
if (configuration?.disabled) {
|
|
48
121
|
return true;
|
|
49
122
|
}
|
|
123
|
+
const normalizedKey = this.#normalizeKey(effectiveKey);
|
|
124
|
+
const permissionMap = this.#permissionMapByKey[normalizedKey] || {};
|
|
50
125
|
const permissions = Array.isArray(permission) ? permission : [permission];
|
|
51
|
-
return permissions.some(
|
|
126
|
+
return permissions.some(val => permissionMap[val]);
|
|
52
127
|
};
|
|
53
|
-
getToken = async () => {
|
|
54
|
-
|
|
128
|
+
getToken = async (key) => {
|
|
129
|
+
const effectiveKey = this.#getEffectivePermissionKey(key);
|
|
130
|
+
const getToken = this.#getConfigurationByKey(effectiveKey)?.getToken;
|
|
131
|
+
if (!getToken) {
|
|
55
132
|
throw new Error('[Permission] Method getToken');
|
|
56
133
|
}
|
|
57
|
-
const token =
|
|
58
|
-
if (token
|
|
59
|
-
return
|
|
134
|
+
const token = await SdResolveMaybeAsync(getToken());
|
|
135
|
+
if (token === '') {
|
|
136
|
+
return undefined;
|
|
60
137
|
}
|
|
61
138
|
return token;
|
|
62
139
|
};
|
|
63
|
-
decodeToken = async () => {
|
|
64
|
-
const token = await this.getToken();
|
|
140
|
+
decodeToken = async (key) => {
|
|
141
|
+
const token = await this.getToken(key);
|
|
65
142
|
if (!token) {
|
|
66
143
|
return null;
|
|
67
144
|
}
|
|
@@ -79,49 +156,46 @@ class SdPermissionService {
|
|
|
79
156
|
return null;
|
|
80
157
|
}
|
|
81
158
|
};
|
|
82
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionService, deps: [
|
|
159
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
83
160
|
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionService, providedIn: 'root' });
|
|
84
161
|
}
|
|
85
162
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionService, decorators: [{
|
|
86
163
|
type: Injectable,
|
|
87
164
|
args: [{ providedIn: 'root' }]
|
|
88
|
-
}], ctorParameters: () => [
|
|
89
|
-
type: Inject,
|
|
90
|
-
args: [SD_PERMISSION_CONFIGURATION]
|
|
91
|
-
}] }, { type: i1.SdCacheService }] });
|
|
165
|
+
}], ctorParameters: () => [] });
|
|
92
166
|
|
|
93
167
|
class SdPermissionDirective {
|
|
94
|
-
templateRef;
|
|
95
|
-
viewContainerRef;
|
|
96
|
-
permissionService;
|
|
168
|
+
#templateRef = inject(TemplateRef);
|
|
169
|
+
#viewContainerRef = inject(ViewContainerRef);
|
|
170
|
+
#permissionService = inject(SdPermissionService);
|
|
97
171
|
// Nếu là mảng thì chỉ cần có 1 permission trong mảng đó xem như có quyền
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
172
|
+
sdPermission = input(undefined);
|
|
173
|
+
sdPermissionKey = input(undefined);
|
|
174
|
+
constructor() {
|
|
175
|
+
effect(() => {
|
|
176
|
+
const permission = this.sdPermission();
|
|
177
|
+
const permissionKey = this.sdPermissionKey();
|
|
178
|
+
this.#viewContainerRef.clear();
|
|
179
|
+
// Nếu không gắn permission thì render
|
|
180
|
+
if (!permission?.toString()) {
|
|
181
|
+
this.#viewContainerRef.createEmbeddedView(this.#templateRef);
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
// Kiểm tra permission theo key (nếu có)
|
|
185
|
+
if (this.#permissionService.hasPermission(permission, permissionKey)) {
|
|
186
|
+
this.#viewContainerRef.createEmbeddedView(this.#templateRef);
|
|
187
|
+
}
|
|
188
|
+
});
|
|
113
189
|
}
|
|
114
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionDirective, deps: [
|
|
115
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "
|
|
190
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
191
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "19.2.17", type: SdPermissionDirective, isStandalone: true, selector: "[sdPermission]", inputs: { sdPermission: { classPropertyName: "sdPermission", publicName: "sdPermission", isSignal: true, isRequired: false, transformFunction: null }, sdPermissionKey: { classPropertyName: "sdPermissionKey", publicName: "sdPermissionKey", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0 });
|
|
116
192
|
}
|
|
117
193
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionDirective, decorators: [{
|
|
118
194
|
type: Directive,
|
|
119
195
|
args: [{
|
|
120
196
|
selector: '[sdPermission]',
|
|
121
197
|
}]
|
|
122
|
-
}], ctorParameters: () => [
|
|
123
|
-
type: Input
|
|
124
|
-
}] } });
|
|
198
|
+
}], ctorParameters: () => [] });
|
|
125
199
|
|
|
126
200
|
class SdPermissionGuard {
|
|
127
201
|
configuration;
|
|
@@ -130,18 +204,30 @@ class SdPermissionGuard {
|
|
|
130
204
|
this.configuration = configuration;
|
|
131
205
|
this.permissionService = permissionService;
|
|
132
206
|
}
|
|
207
|
+
#getConfigurations = () => {
|
|
208
|
+
const config = this.configuration;
|
|
209
|
+
if (!config) {
|
|
210
|
+
return [];
|
|
211
|
+
}
|
|
212
|
+
return Array.isArray(config) ? config : [config];
|
|
213
|
+
};
|
|
133
214
|
canActivate = async (next, state) => {
|
|
134
|
-
//
|
|
135
|
-
await this.permissionService.
|
|
215
|
+
// Guard ở layer portal: preload toàn bộ permission theo tất cả key đã cấu hình
|
|
216
|
+
await this.permissionService.loadAllPermissions().catch(console.error);
|
|
136
217
|
return true;
|
|
137
218
|
};
|
|
138
219
|
canActivateChild = async (activatedRoute, state) => {
|
|
139
220
|
const permission = activatedRoute.data['permission'];
|
|
140
|
-
|
|
221
|
+
const permissionKey = activatedRoute.data?.['permissionKey'];
|
|
222
|
+
if (this.permissionService.hasPermission(permission, permissionKey)) {
|
|
141
223
|
return true;
|
|
142
224
|
}
|
|
143
|
-
|
|
144
|
-
|
|
225
|
+
const configurations = this.#getConfigurations();
|
|
226
|
+
const onForbiden = configurations
|
|
227
|
+
.filter(config => config.key === permissionKey || (permissionKey !== undefined && config.key === undefined))
|
|
228
|
+
.map(config => config.onForbiden)
|
|
229
|
+
.find(val => !!val);
|
|
230
|
+
onForbiden?.();
|
|
145
231
|
return false;
|
|
146
232
|
};
|
|
147
233
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.17", ngImport: i0, type: SdPermissionGuard, deps: [{ token: SD_PERMISSION_CONFIGURATION }, { token: SdPermissionService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"sd-angular-core-modules-permission.mjs","sources":["../../../projects/sd-angular/modules/permission/src/configurations/permission.configuration.ts","../../../projects/sd-angular/modules/permission/src/services/permission.service.ts","../../../projects/sd-angular/modules/permission/src/directives/permission.directive.ts","../../../projects/sd-angular/modules/permission/src/guards/permission.guard.ts","../../../projects/sd-angular/modules/permission/sd-angular-core-modules-permission.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\r\nimport { SdMaybeAsync } from '@sd-angular/core/utilities';\r\n\r\nexport interface ISdPermissionConfiguration {\r\n disabled?: boolean; // Vô hiệu việc kiểm tra permission, mục đích để test chức năng khi chưa có permisison\r\n loadPermissions: () => SdMaybeAsync<string[]>;\r\n onForbiden?: () => void; // Xử lý khi không có quyền truy cập vào URL\r\n getToken?: () => Promise<string | undefined | null> | string | undefined | null; // Lấy thông tin token\r\n}\r\n\r\nexport const SD_PERMISSION_CONFIGURATION = new InjectionToken<ISdPermissionConfiguration>('sd-permission.configuration');\r\n","import { Inject, Injectable } from '@angular/core';\r\nimport { SdCache, SdCacheService } from '@sd-angular/core/services/cache';\r\nimport { ArrayUtilities } from '@sd-angular/core/utilities/extensions';\r\nimport { ISdPermissionConfiguration, SD_PERMISSION_CONFIGURATION } from '../configurations';\r\nimport { SdResolveMaybeAsync } from '@sd-angular/core/utilities';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class SdPermissionService {\r\n #permission: Record<string, boolean> = {};\r\n #permissions: SdCache<string[]>;\r\n #loaded = false;\r\n constructor(\r\n @Inject(SD_PERMISSION_CONFIGURATION) private configuration: ISdPermissionConfiguration,\r\n private cacheService: SdCacheService\r\n ) {\r\n this.#permissions = this.cacheService.create<string[]>('212a51fa-38d5-43b2-bd46-922d85950ba3', {\r\n type: 'session',\r\n default: [],\r\n });\r\n }\r\n\r\n loadPermissions = async () => {\r\n if (!this.#loaded) {\r\n try {\r\n // Trả về Promise<string[]> nhưng chấp nhận mọi kiểu MaybeAsync\r\n const permissions: string[] = await SdResolveMaybeAsync(this.configuration.loadPermissions());\r\n const distinctPermissions = ArrayUtilities.distinct(permissions);\r\n this.#permissions.set(ArrayUtilities.distinct(permissions));\r\n // Chuyển sang dạng map để kiểm tra nhanh hơn\r\n this.#permission = {};\r\n distinctPermissions.forEach(p => (this.#permission[p] = true));\r\n this.#loaded = true;\r\n } catch (err) {\r\n console.error(err);\r\n this.#permissions.set([]);\r\n this.#permission = {};\r\n this.#loaded = true;\r\n }\r\n }\r\n };\r\n\r\n hasPermission = (permission: string | string[]) => {\r\n if (!permission?.toString()) {\r\n return true;\r\n }\r\n if (this.configuration.disabled) {\r\n return true;\r\n }\r\n const permissions = Array.isArray(permission) ? permission : [permission];\r\n return permissions.some(permission => this.#permission[permission]);\r\n };\r\n\r\n getToken = async () => {\r\n if (!this.configuration.getToken) {\r\n throw new Error('[Permission] Method getToken');\r\n }\r\n const token = this.configuration.getToken();\r\n if (token instanceof Promise) {\r\n return await token;\r\n }\r\n return token;\r\n };\r\n\r\n decodeToken = async <T>(): Promise<T | null> => {\r\n const token = await this.getToken();\r\n if (!token) {\r\n return null;\r\n }\r\n try {\r\n const payload = token.split('.')[1];\r\n const base64 = payload.replace(/-/g, '+').replace(/_/g, '/');\r\n const jsonPayload = decodeURIComponent(\r\n atob(base64)\r\n .split('')\r\n .map(c => '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2))\r\n .join('')\r\n );\r\n return JSON.parse(jsonPayload);\r\n } catch (error) {\r\n console.error('Invalid token', error);\r\n return null;\r\n }\r\n };\r\n}\r\n","import { Directive, Input, TemplateRef, ViewContainerRef } from '@angular/core';\r\nimport { SdPermissionService } from '../services';\r\n\r\n@Directive({\r\n selector: '[sdPermission]',\r\n})\r\nexport class SdPermissionDirective {\r\n // Nếu là mảng thì chỉ cần có 1 permission trong mảng đó xem như có quyền\r\n @Input() set sdPermission(permission: string | string[] | undefined | null) {\r\n // Nếu không gắn permisison thì render\r\n if (!permission?.toString()) {\r\n this.viewContainerRef.createEmbeddedView(this.templateRef);\r\n return;\r\n }\r\n // Kiểm tra permisison\r\n if (this.permissionService.hasPermission(permission)) {\r\n this.viewContainerRef.createEmbeddedView(this.templateRef);\r\n }\r\n }\r\n\r\n constructor(\r\n private templateRef: TemplateRef<any>,\r\n private viewContainerRef: ViewContainerRef,\r\n private permissionService: SdPermissionService\r\n ) {}\r\n}\r\n","import { Inject, Injectable } from '@angular/core';\r\nimport { ActivatedRouteSnapshot, CanActivate, CanActivateChild, RouterStateSnapshot } from '@angular/router';\r\nimport { ISdPermissionConfiguration, SD_PERMISSION_CONFIGURATION } from '../configurations';\r\nimport { SdPermissionService } from '../services';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class SdPermissionGuard implements CanActivate, CanActivateChild {\r\n constructor(\r\n @Inject(SD_PERMISSION_CONFIGURATION)\r\n private configuration: ISdPermissionConfiguration,\r\n private permissionService: SdPermissionService\r\n ) {}\r\n\r\n canActivate = async (next: ActivatedRouteSnapshot, state: RouterStateSnapshot) => {\r\n // Đảm bảo permission đã được loaded\r\n await this.permissionService.loadPermissions().catch(console.error);\r\n return true;\r\n };\r\n\r\n canActivateChild = async (activatedRoute: ActivatedRouteSnapshot, state: RouterStateSnapshot) => {\r\n const permission = activatedRoute.data['permission'];\r\n if (this.permissionService.hasPermission(permission)) {\r\n return true;\r\n }\r\n //\r\n this.configuration?.onForbiden?.();\r\n return false;\r\n };\r\n}\r\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":["i1.SdPermissionService"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"sd-angular-core-modules-permission.mjs","sources":["../../../projects/sd-angular/modules/permission/src/configurations/permission.configuration.ts","../../../projects/sd-angular/modules/permission/src/services/permission.service.ts","../../../projects/sd-angular/modules/permission/src/directives/permission.directive.ts","../../../projects/sd-angular/modules/permission/src/guards/permission.guard.ts","../../../projects/sd-angular/modules/permission/sd-angular-core-modules-permission.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\r\nimport { SdMaybeAsync } from '@sd-angular/core/utilities';\r\n\r\n/**\r\n * Cấu hình trung tâm cho module permission.\r\n *\r\n * Luồng hoạt động chính:\r\n * 1. `loadPermissions(key?)` ở service được gọi để lấy danh sách mã quyền theo từng `key` cấu hình.\r\n * 2. Guard/service đối chiếu mã quyền theo route metadata.\r\n * 3. Khi không đủ quyền, callback `onForbiden()` sẽ được gọi (nếu có).\r\n *\r\n * Lưu ý:\r\n * - `SD_PERMISSION_CONFIGURATION` hỗ trợ cả cấu hình đơn và mảng cấu hình (`multi: true`).\r\n * - Nếu cần tạm bỏ qua kiểm tra quyền (POC/UAT cục bộ), đặt `disabled = true`.\r\n */\r\nexport interface ISdPermissionConfiguration {\r\n /**\r\n * Khóa định danh cấu hình.\r\n * Dùng để phân biệt khi hệ thống mở rộng theo nhiều profile permission.\r\n *\r\n * Lưu ý: `undefined` cũng được xem là một key hợp lệ (cấu hình mặc định).\r\n */\r\n key?: string;\r\n\r\n /**\r\n * Bật/tắt kiểm tra permission toàn cục.\r\n * - `true`: bỏ qua kiểm tra quyền.\r\n * - `false | undefined`: kiểm tra quyền theo cấu hình route.\r\n */\r\n disabled?: boolean;\r\n\r\n /**\r\n * Trả về danh sách mã quyền của user hiện tại.\r\n * Có thể đồng bộ hoặc bất đồng bộ.\r\n *\r\n * Ví dụ giá trị trả về:\r\n * - `['PRODUCT_C_EMPLOYEE_VIEW', 'PRODUCT_C_EMPLOYEE_UPDATE']`\r\n */\r\n loadPermissions: () => SdMaybeAsync<string[]>;\r\n\r\n /**\r\n * Callback xử lý khi user không có quyền truy cập URL hiện tại.\r\n * Thường dùng để điều hướng sang trang forbidden hoặc hiển thị thông báo.\r\n *\r\n * Giữ nguyên tên `onForbiden` để tương thích API hiện tại.\r\n */\r\n onForbiden?: () => void;\r\n\r\n /**\r\n * Cung cấp access token hiện tại cho các tác vụ liên quan permission.\r\n * Hỗ trợ trả về đồng bộ, Promise hoặc Observable.\r\n */\r\n getToken?: () => SdMaybeAsync<string | undefined | null>;\r\n}\r\n\r\n/**\r\n * InjectionToken cho cấu hình permission.\r\n *\r\n * Ví dụ provider:\r\n * {\r\n * provide: SD_PERMISSION_CONFIGURATION,\r\n * useValue: {\r\n * disabled: false,\r\n * loadPermissions: () => ['SAMPLE_C_EMPLOYEE_VIEW'],\r\n * onForbiden: () => router.navigateByUrl('/layout/forbidden')\r\n * }\r\n * }\r\n */\r\nexport const SD_PERMISSION_CONFIGURATION =\r\n new InjectionToken<ISdPermissionConfiguration | ISdPermissionConfiguration[]>('sd-permission.configuration');\r\n","import { inject, Injectable } from '@angular/core';\r\nimport { SdCache, SdCacheService } from '@sd-angular/core/services/cache';\r\nimport { ArrayUtilities } from '@sd-angular/core/utilities/extensions';\r\nimport { ISdPermissionConfiguration, SD_PERMISSION_CONFIGURATION } from '../configurations';\r\nimport { SdMaybeAsync, SdResolveMaybeAsync } from '@sd-angular/core/utilities';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class SdPermissionService {\r\n #permissionMapByKey: Record<string, Record<string, boolean>> = {};\r\n readonly #configuration = inject<ISdPermissionConfiguration | ISdPermissionConfiguration[]>(SD_PERMISSION_CONFIGURATION);\r\n readonly #cacheService = inject(SdCacheService);\r\n #permissionsByKey: SdCache<Record<string, string[]>> = this.#cacheService.create<Record<string, string[]>>('212a51fa-38d5-43b2-bd46-922d85950ba3', {\r\n type: 'session',\r\n default: {},\r\n });\r\n readonly #loadedKeys = new Set<string>();\r\n\r\n constructor() {\r\n this.#validateDuplicateConfigKeys();\r\n }\r\n\r\n #getConfigurations = (): ISdPermissionConfiguration[] => {\r\n const config = this.#configuration;\r\n if (!config) {\r\n return [];\r\n }\r\n return Array.isArray(config) ? config : [config];\r\n };\r\n\r\n #normalizeKey = (key?: string): string => {\r\n return key === undefined ? '__undefined__' : key;\r\n };\r\n\r\n #validateDuplicateConfigKeys = (): void => {\r\n const seen = new Set<string>();\r\n for (const config of this.#getConfigurations()) {\r\n const normalizedKey = this.#normalizeKey(config.key);\r\n if (seen.has(normalizedKey)) {\r\n const keyLabel = config.key === undefined ? 'undefined' : config.key;\r\n throw new Error(`[Permission] Duplicate permission configuration key: ${keyLabel}`);\r\n }\r\n seen.add(normalizedKey);\r\n }\r\n };\r\n\r\n #getConfigurationByKey = (key?: string): ISdPermissionConfiguration | undefined => {\r\n return this.#getConfigurations().find(config => config.key === key);\r\n };\r\n\r\n #getEffectivePermissionKey = (key?: string): string | undefined => {\r\n if (this.#getConfigurationByKey(key)) {\r\n return key;\r\n }\r\n\r\n // Portal-level config uses key = undefined and acts as default fallback\r\n if (key !== undefined && this.#getConfigurationByKey(undefined)) {\r\n return undefined;\r\n }\r\n\r\n return key;\r\n };\r\n\r\n #setPermissionsForKey = (normalizedKey: string, permissions: string[]): void => {\r\n const distinctPermissions = ArrayUtilities.distinct(permissions || []);\r\n const current = this.#permissionsByKey.get() || {};\r\n this.#permissionsByKey.set({\r\n ...current,\r\n [normalizedKey]: distinctPermissions,\r\n });\r\n\r\n const permissionMap: Record<string, boolean> = {};\r\n distinctPermissions.forEach(permission => {\r\n permissionMap[permission] = true;\r\n });\r\n this.#permissionMapByKey[normalizedKey] = permissionMap;\r\n };\r\n\r\n loadPermissions = async (key?: string): Promise<string[]> => {\r\n const effectiveKey = this.#getEffectivePermissionKey(key);\r\n const normalizedKey = this.#normalizeKey(effectiveKey);\r\n if (this.#loadedKeys.has(normalizedKey)) {\r\n return this.#permissionsByKey.get()?.[normalizedKey] || [];\r\n }\r\n\r\n const configuration = this.#getConfigurationByKey(effectiveKey);\r\n if (!configuration) {\r\n this.#setPermissionsForKey(normalizedKey, []);\r\n this.#loadedKeys.add(normalizedKey);\r\n return [];\r\n }\r\n\r\n try {\r\n const permissions: string[] = await SdResolveMaybeAsync(configuration.loadPermissions());\r\n this.#setPermissionsForKey(normalizedKey, permissions || []);\r\n } catch (err) {\r\n console.error(err);\r\n this.#setPermissionsForKey(normalizedKey, []);\r\n } finally {\r\n this.#loadedKeys.add(normalizedKey);\r\n }\r\n\r\n return this.#permissionsByKey.get()?.[normalizedKey] || [];\r\n };\r\n\r\n loadAllPermissions = async (): Promise<void> => {\r\n const configurations = this.#getConfigurations();\r\n if (!configurations.length) {\r\n await this.loadPermissions(undefined);\r\n return;\r\n }\r\n\r\n await Promise.all(configurations.map(config => this.loadPermissions(config.key)));\r\n };\r\n\r\n hasPermission = (permission: string | string[], key?: string) => {\r\n if (!permission?.toString()) {\r\n return true;\r\n }\r\n\r\n const effectiveKey = this.#getEffectivePermissionKey(key);\r\n const configuration = this.#getConfigurationByKey(effectiveKey);\r\n if (configuration?.disabled) {\r\n return true;\r\n }\r\n\r\n const normalizedKey = this.#normalizeKey(effectiveKey);\r\n const permissionMap = this.#permissionMapByKey[normalizedKey] || {};\r\n const permissions = Array.isArray(permission) ? permission : [permission];\r\n return permissions.some(val => permissionMap[val]);\r\n };\r\n\r\n getToken = async (key?: string) => {\r\n const effectiveKey = this.#getEffectivePermissionKey(key);\r\n const getToken = this.#getConfigurationByKey(effectiveKey)?.getToken as (() => SdMaybeAsync<string | undefined | null>) | undefined;\r\n if (!getToken) {\r\n throw new Error('[Permission] Method getToken');\r\n }\r\n\r\n const token = await SdResolveMaybeAsync(getToken());\r\n if (token === '') {\r\n return undefined;\r\n }\r\n return token;\r\n };\r\n\r\n decodeToken = async <T>(key?: string): Promise<T | null> => {\r\n const token = await this.getToken(key);\r\n if (!token) {\r\n return null;\r\n }\r\n try {\r\n const payload = token.split('.')[1];\r\n const base64 = payload.replace(/-/g, '+').replace(/_/g, '/');\r\n const jsonPayload = decodeURIComponent(\r\n atob(base64)\r\n .split('')\r\n .map(c => '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2))\r\n .join('')\r\n );\r\n return JSON.parse(jsonPayload);\r\n } catch (error) {\r\n console.error('Invalid token', error);\r\n return null;\r\n }\r\n };\r\n}\r\n","import { Directive, effect, inject, input, TemplateRef, ViewContainerRef } from '@angular/core';\r\nimport { SdPermissionService } from '../services';\r\n\r\n@Directive({\r\n selector: '[sdPermission]',\r\n})\r\nexport class SdPermissionDirective {\r\n readonly #templateRef = inject<TemplateRef<any>>(TemplateRef);\r\n readonly #viewContainerRef = inject(ViewContainerRef);\r\n readonly #permissionService = inject(SdPermissionService);\r\n\r\n // Nếu là mảng thì chỉ cần có 1 permission trong mảng đó xem như có quyền\r\n readonly sdPermission = input<string | string[] | undefined | null>(undefined);\r\n readonly sdPermissionKey = input<string | undefined>(undefined);\r\n\r\n constructor() {\r\n effect(() => {\r\n const permission = this.sdPermission();\r\n const permissionKey = this.sdPermissionKey();\r\n\r\n this.#viewContainerRef.clear();\r\n\r\n // Nếu không gắn permission thì render\r\n if (!permission?.toString()) {\r\n this.#viewContainerRef.createEmbeddedView(this.#templateRef);\r\n return;\r\n }\r\n\r\n // Kiểm tra permission theo key (nếu có)\r\n if (this.#permissionService.hasPermission(permission, permissionKey)) {\r\n this.#viewContainerRef.createEmbeddedView(this.#templateRef);\r\n }\r\n });\r\n }\r\n}\r\n","import { Inject, Injectable } from '@angular/core';\r\nimport { ActivatedRouteSnapshot, CanActivate, CanActivateChild, RouterStateSnapshot } from '@angular/router';\r\nimport { ISdPermissionConfiguration, SD_PERMISSION_CONFIGURATION } from '../configurations';\r\nimport { SdPermissionService } from '../services';\r\n\r\n@Injectable({ providedIn: 'root' })\r\nexport class SdPermissionGuard implements CanActivate, CanActivateChild {\r\n constructor(\r\n @Inject(SD_PERMISSION_CONFIGURATION)\r\n private configuration: ISdPermissionConfiguration | ISdPermissionConfiguration[],\r\n private permissionService: SdPermissionService\r\n ) {}\r\n\r\n #getConfigurations = (): ISdPermissionConfiguration[] => {\r\n const config = this.configuration;\r\n if (!config) {\r\n return [];\r\n }\r\n return Array.isArray(config) ? config : [config];\r\n };\r\n\r\n canActivate = async (next: ActivatedRouteSnapshot, state: RouterStateSnapshot) => {\r\n // Guard ở layer portal: preload toàn bộ permission theo tất cả key đã cấu hình\r\n await this.permissionService.loadAllPermissions().catch(console.error);\r\n return true;\r\n };\r\n\r\n canActivateChild = async (activatedRoute: ActivatedRouteSnapshot, state: RouterStateSnapshot) => {\r\n const permission = activatedRoute.data['permission'];\r\n const permissionKey = activatedRoute.data?.['permissionKey'] as string | undefined;\r\n if (this.permissionService.hasPermission(permission, permissionKey)) {\r\n return true;\r\n }\r\n\r\n const configurations = this.#getConfigurations();\r\n const onForbiden = configurations\r\n .filter(config => config.key === permissionKey || (permissionKey !== undefined && config.key === undefined))\r\n .map(config => config.onForbiden)\r\n .find(val => !!val);\r\n onForbiden?.();\r\n return false;\r\n };\r\n}\r\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":["i1.SdPermissionService"],"mappings":";;;;;;AAuDA;;;;;;;;;;;;AAYG;MACU,2BAA2B,GACtC,IAAI,cAAc,CAA4D,6BAA6B;;MC9DhG,mBAAmB,CAAA;IAC9B,mBAAmB,GAA4C,EAAE;AACxD,IAAA,cAAc,GAAG,MAAM,CAA4D,2BAA2B,CAAC;AAC/G,IAAA,aAAa,GAAG,MAAM,CAAC,cAAc,CAAC;IAC/C,iBAAiB,GAAsC,IAAI,CAAC,aAAa,CAAC,MAAM,CAA2B,sCAAsC,EAAE;AACjJ,QAAA,IAAI,EAAE,SAAS;AACf,QAAA,OAAO,EAAE,EAAE;AACZ,KAAA,CAAC;AACO,IAAA,WAAW,GAAG,IAAI,GAAG,EAAU;AAExC,IAAA,WAAA,GAAA;QACE,IAAI,CAAC,4BAA4B,EAAE;IACrC;IAEA,kBAAkB,GAAG,MAAmC;AACtD,QAAA,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc;QAClC,IAAI,CAAC,MAAM,EAAE;AACX,YAAA,OAAO,EAAE;QACX;AACA,QAAA,OAAO,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC;AAClD,IAAA,CAAC;AAED,IAAA,aAAa,GAAG,CAAC,GAAY,KAAY;QACvC,OAAO,GAAG,KAAK,SAAS,GAAG,eAAe,GAAG,GAAG;AAClD,IAAA,CAAC;IAED,4BAA4B,GAAG,MAAW;AACxC,QAAA,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU;QAC9B,KAAK,MAAM,MAAM,IAAI,IAAI,CAAC,kBAAkB,EAAE,EAAE;YAC9C,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,GAAG,CAAC;AACpD,YAAA,IAAI,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE;AAC3B,gBAAA,MAAM,QAAQ,GAAG,MAAM,CAAC,GAAG,KAAK,SAAS,GAAG,WAAW,GAAG,MAAM,CAAC,GAAG;AACpE,gBAAA,MAAM,IAAI,KAAK,CAAC,wDAAwD,QAAQ,CAAA,CAAE,CAAC;YACrF;AACA,YAAA,IAAI,CAAC,GAAG,CAAC,aAAa,CAAC;QACzB;AACF,IAAA,CAAC;AAED,IAAA,sBAAsB,GAAG,CAAC,GAAY,KAA4C;AAChF,QAAA,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC,IAAI,CAAC,MAAM,IAAI,MAAM,CAAC,GAAG,KAAK,GAAG,CAAC;AACrE,IAAA,CAAC;AAED,IAAA,0BAA0B,GAAG,CAAC,GAAY,KAAwB;AAChE,QAAA,IAAI,IAAI,CAAC,sBAAsB,CAAC,GAAG,CAAC,EAAE;AACpC,YAAA,OAAO,GAAG;QACZ;;QAGA,IAAI,GAAG,KAAK,SAAS,IAAI,IAAI,CAAC,sBAAsB,CAAC,SAAS,CAAC,EAAE;AAC/D,YAAA,OAAO,SAAS;QAClB;AAEA,QAAA,OAAO,GAAG;AACZ,IAAA,CAAC;AAED,IAAA,qBAAqB,GAAG,CAAC,aAAqB,EAAE,WAAqB,KAAU;QAC7E,MAAM,mBAAmB,GAAG,cAAc,CAAC,QAAQ,CAAC,WAAW,IAAI,EAAE,CAAC;QACtE,MAAM,OAAO,GAAG,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,IAAI,EAAE;AAClD,QAAA,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC;AACzB,YAAA,GAAG,OAAO;YACV,CAAC,aAAa,GAAG,mBAAmB;AACrC,SAAA,CAAC;QAEF,MAAM,aAAa,GAA4B,EAAE;AACjD,QAAA,mBAAmB,CAAC,OAAO,CAAC,UAAU,IAAG;AACvC,YAAA,aAAa,CAAC,UAAU,CAAC,GAAG,IAAI;AAClC,QAAA,CAAC,CAAC;AACF,QAAA,IAAI,CAAC,mBAAmB,CAAC,aAAa,CAAC,GAAG,aAAa;AACzD,IAAA,CAAC;AAED,IAAA,eAAe,GAAG,OAAO,GAAY,KAAuB;QAC1D,MAAM,YAAY,GAAG,IAAI,CAAC,0BAA0B,CAAC,GAAG,CAAC;QACzD,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC;QACtD,IAAI,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE;AACvC,YAAA,OAAO,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,GAAG,aAAa,CAAC,IAAI,EAAE;QAC5D;QAEA,MAAM,aAAa,GAAG,IAAI,CAAC,sBAAsB,CAAC,YAAY,CAAC;QAC/D,IAAI,CAAC,aAAa,EAAE;AAClB,YAAA,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,EAAE,CAAC;AAC7C,YAAA,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,aAAa,CAAC;AACnC,YAAA,OAAO,EAAE;QACX;AAEA,QAAA,IAAI;YACF,MAAM,WAAW,GAAa,MAAM,mBAAmB,CAAC,aAAa,CAAC,eAAe,EAAE,CAAC;YACxF,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,WAAW,IAAI,EAAE,CAAC;QAC9D;QAAE,OAAO,GAAG,EAAE;AACZ,YAAA,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC;AAClB,YAAA,IAAI,CAAC,qBAAqB,CAAC,aAAa,EAAE,EAAE,CAAC;QAC/C;gBAAU;AACR,YAAA,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,aAAa,CAAC;QACrC;AAEA,QAAA,OAAO,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,GAAG,aAAa,CAAC,IAAI,EAAE;AAC5D,IAAA,CAAC;IAED,kBAAkB,GAAG,YAA0B;AAC7C,QAAA,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,EAAE;AAChD,QAAA,IAAI,CAAC,cAAc,CAAC,MAAM,EAAE;AAC1B,YAAA,MAAM,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;YACrC;QACF;QAEA,MAAM,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,GAAG,CAAC,MAAM,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;AACnF,IAAA,CAAC;AAED,IAAA,aAAa,GAAG,CAAC,UAA6B,EAAE,GAAY,KAAI;AAC9D,QAAA,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,EAAE;AAC3B,YAAA,OAAO,IAAI;QACb;QAEA,MAAM,YAAY,GAAG,IAAI,CAAC,0BAA0B,CAAC,GAAG,CAAC;QACzD,MAAM,aAAa,GAAG,IAAI,CAAC,sBAAsB,CAAC,YAAY,CAAC;AAC/D,QAAA,IAAI,aAAa,EAAE,QAAQ,EAAE;AAC3B,YAAA,OAAO,IAAI;QACb;QAEA,MAAM,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC;QACtD,MAAM,aAAa,GAAG,IAAI,CAAC,mBAAmB,CAAC,aAAa,CAAC,IAAI,EAAE;AACnE,QAAA,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,GAAG,UAAU,GAAG,CAAC,UAAU,CAAC;AACzE,QAAA,OAAO,WAAW,CAAC,IAAI,CAAC,GAAG,IAAI,aAAa,CAAC,GAAG,CAAC,CAAC;AACpD,IAAA,CAAC;AAED,IAAA,QAAQ,GAAG,OAAO,GAAY,KAAI;QAChC,MAAM,YAAY,GAAG,IAAI,CAAC,0BAA0B,CAAC,GAAG,CAAC;QACzD,MAAM,QAAQ,GAAG,IAAI,CAAC,sBAAsB,CAAC,YAAY,CAAC,EAAE,QAAuE;QACnI,IAAI,CAAC,QAAQ,EAAE;AACb,YAAA,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC;QACjD;QAEA,MAAM,KAAK,GAAG,MAAM,mBAAmB,CAAC,QAAQ,EAAE,CAAC;AACnD,QAAA,IAAI,KAAK,KAAK,EAAE,EAAE;AAChB,YAAA,OAAO,SAAS;QAClB;AACA,QAAA,OAAO,KAAK;AACd,IAAA,CAAC;AAED,IAAA,WAAW,GAAG,OAAU,GAAY,KAAuB;QACzD,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC;QACtC,IAAI,CAAC,KAAK,EAAE;AACV,YAAA,OAAO,IAAI;QACb;AACA,QAAA,IAAI;YACF,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;AACnC,YAAA,MAAM,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC;AAC5D,YAAA,MAAM,WAAW,GAAG,kBAAkB,CACpC,IAAI,CAAC,MAAM;iBACR,KAAK,CAAC,EAAE;AACR,iBAAA,GAAG,CAAC,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,GAAG,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC;AAC9D,iBAAA,IAAI,CAAC,EAAE,CAAC,CACZ;AACD,YAAA,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC;QAChC;QAAE,OAAO,KAAK,EAAE;AACd,YAAA,OAAO,CAAC,KAAK,CAAC,eAAe,EAAE,KAAK,CAAC;AACrC,YAAA,OAAO,IAAI;QACb;AACF,IAAA,CAAC;wGA7JU,mBAAmB,EAAA,IAAA,EAAA,EAAA,EAAA,MAAA,EAAA,EAAA,CAAA,eAAA,CAAA,UAAA,EAAA,CAAA;AAAnB,IAAA,OAAA,KAAA,GAAA,EAAA,CAAA,qBAAA,CAAA,EAAA,UAAA,EAAA,QAAA,EAAA,OAAA,EAAA,SAAA,EAAA,QAAA,EAAA,EAAA,EAAA,IAAA,EAAA,mBAAmB,cADN,MAAM,EAAA,CAAA;;4FACnB,mBAAmB,EAAA,UAAA,EAAA,CAAA;kBAD/B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;;MCArB,qBAAqB,CAAA;AACvB,IAAA,YAAY,GAAG,MAAM,CAAmB,WAAW,CAAC;AACpD,IAAA,iBAAiB,GAAG,MAAM,CAAC,gBAAgB,CAAC;AAC5C,IAAA,kBAAkB,GAAG,MAAM,CAAC,mBAAmB,CAAC;;AAGhD,IAAA,YAAY,GAAG,KAAK,CAAuC,SAAS,CAAC;AACrE,IAAA,eAAe,GAAG,KAAK,CAAqB,SAAS,CAAC;AAE/D,IAAA,WAAA,GAAA;QACE,MAAM,CAAC,MAAK;AACV,YAAA,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,EAAE;AACtC,YAAA,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,EAAE;AAE5C,YAAA,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE;;AAG9B,YAAA,IAAI,CAAC,UAAU,EAAE,QAAQ,EAAE,EAAE;gBAC3B,IAAI,CAAC,iBAAiB,CAAC,kBAAkB,CAAC,IAAI,CAAC,YAAY,CAAC;gBAC5D;YACF;;YAGA,IAAI,IAAI,CAAC,kBAAkB,CAAC,aAAa,CAAC,UAAU,EAAE,aAAa,CAAC,EAAE;gBACpE,IAAI,CAAC,iBAAiB,CAAC,kBAAkB,CAAC,IAAI,CAAC,YAAY,CAAC;YAC9D;AACF,QAAA,CAAC,CAAC;IACJ;wGA3BW,qBAAqB,EAAA,IAAA,EAAA,EAAA,EAAA,MAAA,EAAA,EAAA,CAAA,eAAA,CAAA,SAAA,EAAA,CAAA;4FAArB,qBAAqB,EAAA,YAAA,EAAA,IAAA,EAAA,QAAA,EAAA,gBAAA,EAAA,MAAA,EAAA,EAAA,YAAA,EAAA,EAAA,iBAAA,EAAA,cAAA,EAAA,UAAA,EAAA,cAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,KAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,eAAA,EAAA,EAAA,iBAAA,EAAA,iBAAA,EAAA,UAAA,EAAA,iBAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,KAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,EAAA,QAAA,EAAA,EAAA,EAAA,CAAA;;4FAArB,qBAAqB,EAAA,UAAA,EAAA,CAAA;kBAHjC,SAAS;AAAC,YAAA,IAAA,EAAA,CAAA;AACT,oBAAA,QAAQ,EAAE,gBAAgB;AAC3B,iBAAA;;;MCCY,iBAAiB,CAAA;AAGlB,IAAA,aAAA;AACA,IAAA,iBAAA;IAHV,WAAA,CAEU,aAAwE,EACxE,iBAAsC,EAAA;QADtC,IAAA,CAAA,aAAa,GAAb,aAAa;QACb,IAAA,CAAA,iBAAiB,GAAjB,iBAAiB;IACxB;IAEH,kBAAkB,GAAG,MAAmC;AACtD,QAAA,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa;QACjC,IAAI,CAAC,MAAM,EAAE;AACX,YAAA,OAAO,EAAE;QACX;AACA,QAAA,OAAO,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC;AAClD,IAAA,CAAC;AAED,IAAA,WAAW,GAAG,OAAO,IAA4B,EAAE,KAA0B,KAAI;;AAE/E,QAAA,MAAM,IAAI,CAAC,iBAAiB,CAAC,kBAAkB,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;AACtE,QAAA,OAAO,IAAI;AACb,IAAA,CAAC;AAED,IAAA,gBAAgB,GAAG,OAAO,cAAsC,EAAE,KAA0B,KAAI;QAC9F,MAAM,UAAU,GAAG,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC;QACpD,MAAM,aAAa,GAAG,cAAc,CAAC,IAAI,GAAG,eAAe,CAAuB;QAClF,IAAI,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,UAAU,EAAE,aAAa,CAAC,EAAE;AACnE,YAAA,OAAO,IAAI;QACb;AAEA,QAAA,MAAM,cAAc,GAAG,IAAI,CAAC,kBAAkB,EAAE;QAChD,MAAM,UAAU,GAAG;aAChB,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,GAAG,KAAK,aAAa,KAAK,aAAa,KAAK,SAAS,IAAI,MAAM,CAAC,GAAG,KAAK,SAAS,CAAC;aAC1G,GAAG,CAAC,MAAM,IAAI,MAAM,CAAC,UAAU;aAC/B,IAAI,CAAC,GAAG,IAAI,CAAC,CAAC,GAAG,CAAC;QACrB,UAAU,IAAI;AACd,QAAA,OAAO,KAAK;AACd,IAAA,CAAC;AAnCU,IAAA,OAAA,IAAA,GAAA,EAAA,CAAA,kBAAA,CAAA,EAAA,UAAA,EAAA,QAAA,EAAA,OAAA,EAAA,SAAA,EAAA,QAAA,EAAA,EAAA,EAAA,IAAA,EAAA,iBAAiB,kBAElB,2BAA2B,EAAA,EAAA,EAAA,KAAA,EAAAA,mBAAA,EAAA,CAAA,EAAA,MAAA,EAAA,EAAA,CAAA,eAAA,CAAA,UAAA,EAAA,CAAA;AAF1B,IAAA,OAAA,KAAA,GAAA,EAAA,CAAA,qBAAA,CAAA,EAAA,UAAA,EAAA,QAAA,EAAA,OAAA,EAAA,SAAA,EAAA,QAAA,EAAA,EAAA,EAAA,IAAA,EAAA,iBAAiB,cADJ,MAAM,EAAA,CAAA;;4FACnB,iBAAiB,EAAA,UAAA,EAAA,CAAA;kBAD7B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE;;0BAG7B,MAAM;2BAAC,2BAA2B;;;ACRvC;;AAEG;;;;"}
|
|
@@ -72,6 +72,13 @@ const format = (template, ...arr) => {
|
|
|
72
72
|
}
|
|
73
73
|
return template;
|
|
74
74
|
};
|
|
75
|
+
const getValueByPath = (entity, key) => {
|
|
76
|
+
let value = entity;
|
|
77
|
+
for (const part of key.split('.')) {
|
|
78
|
+
value = value?.[part];
|
|
79
|
+
}
|
|
80
|
+
return value;
|
|
81
|
+
};
|
|
75
82
|
// Convert: `[${code}] ${name}` => `[Mã] Tên`;
|
|
76
83
|
const templateToDisplay = (template, entity) => {
|
|
77
84
|
if (!template) {
|
|
@@ -82,54 +89,39 @@ const templateToDisplay = (template, entity) => {
|
|
|
82
89
|
for (const match of matches) {
|
|
83
90
|
const key = match.slice(2, match.length - 1);
|
|
84
91
|
if (key) {
|
|
85
|
-
template = template.replace(match, entity
|
|
92
|
+
template = template.replace(match, getValueByPath(entity, key) ?? '');
|
|
86
93
|
}
|
|
87
94
|
}
|
|
88
95
|
return template;
|
|
89
96
|
};
|
|
90
|
-
|
|
97
|
+
const EXACT_TEMPLATE_REGEX = /^\$\{([A-Za-z0-9._-]*)\}$/;
|
|
98
|
+
const NUMBER_LITERAL_REGEX = /^-?\d+(\.\d+)?$/;
|
|
99
|
+
// Convert placeholder/literal an toàn, không thực thi JavaScript động.
|
|
91
100
|
const parseExpression = (template, entity) => {
|
|
92
101
|
if (!template) {
|
|
93
102
|
return undefined;
|
|
94
103
|
}
|
|
95
|
-
const
|
|
96
|
-
const
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
if (key) {
|
|
100
|
-
// Xử lý trong trường hợp key có định dạng a.b.c ...
|
|
101
|
-
let val = entity;
|
|
102
|
-
const strs = key.split('.');
|
|
103
|
-
for (const str of strs) {
|
|
104
|
-
val = val?.[str];
|
|
105
|
-
}
|
|
106
|
-
if (typeof val === 'string') {
|
|
107
|
-
// Nếu là chuỗi thì thêm dấu nháy khi replace giá trị để so sánh, ví dụ 'a' === 'a'
|
|
108
|
-
// Nếu là số hay boolean thì không cần, ví dụ 121 === 121
|
|
109
|
-
template = template.replace(match, `'${val}'`);
|
|
110
|
-
}
|
|
111
|
-
else {
|
|
112
|
-
// Nếu là object/array thì phải dùng JSON.stringify
|
|
113
|
-
if (typeof val === 'object' && !!val) {
|
|
114
|
-
template = template.replace(match, JSON.stringify(val) ?? `undefined`);
|
|
115
|
-
}
|
|
116
|
-
else {
|
|
117
|
-
template = template.replace(match, val ?? `undefined`);
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
}
|
|
104
|
+
const trimmedTemplate = template.trim();
|
|
105
|
+
const exactTemplateMatch = trimmedTemplate.match(EXACT_TEMPLATE_REGEX);
|
|
106
|
+
if (exactTemplateMatch?.[1]) {
|
|
107
|
+
return getValueByPath(entity, exactTemplateMatch[1]);
|
|
121
108
|
}
|
|
122
|
-
if (
|
|
123
|
-
return
|
|
109
|
+
if (trimmedTemplate === 'true') {
|
|
110
|
+
return true;
|
|
124
111
|
}
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
return result;
|
|
112
|
+
if (trimmedTemplate === 'false') {
|
|
113
|
+
return false;
|
|
128
114
|
}
|
|
129
|
-
|
|
130
|
-
|
|
115
|
+
if (trimmedTemplate === 'null') {
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
if (trimmedTemplate === 'undefined') {
|
|
131
119
|
return undefined;
|
|
132
120
|
}
|
|
121
|
+
if (NUMBER_LITERAL_REGEX.test(trimmedTemplate)) {
|
|
122
|
+
return Number(trimmedTemplate);
|
|
123
|
+
}
|
|
124
|
+
return templateToDisplay(template, entity);
|
|
133
125
|
};
|
|
134
126
|
const SALT = 'cb9f4b2a-d26c-4787-a66e-e7130ee00f95';
|
|
135
127
|
const encrypt = (obj) => {
|