velocious 1.0.201 → 1.0.203
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/README.md +88 -0
- package/build/src/authorization/ability.d.ts +167 -0
- package/build/src/authorization/ability.d.ts.map +1 -0
- package/build/src/authorization/ability.js +216 -0
- package/build/src/authorization/base-resource.d.ts +58 -0
- package/build/src/authorization/base-resource.d.ts.map +1 -0
- package/build/src/authorization/base-resource.js +66 -0
- package/build/src/cli/commands/db/schema/dump.d.ts +2 -10
- package/build/src/cli/commands/db/schema/dump.d.ts.map +1 -1
- package/build/src/cli/commands/db/schema/dump.js +3 -29
- package/build/src/cli/commands/generate/frontend-models.d.ts +7 -0
- package/build/src/cli/commands/generate/frontend-models.d.ts.map +1 -0
- package/build/src/cli/commands/generate/frontend-models.js +9 -0
- package/build/src/configuration-types.d.ts +73 -0
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +23 -1
- package/build/src/configuration.d.ts +45 -2
- package/build/src/configuration.d.ts.map +1 -1
- package/build/src/configuration.js +60 -2
- package/build/src/controller.d.ts +2 -0
- package/build/src/controller.d.ts.map +1 -1
- package/build/src/controller.js +5 -1
- package/build/src/database/record/index.d.ts +14 -0
- package/build/src/database/record/index.d.ts.map +1 -1
- package/build/src/database/record/index.js +28 -1
- package/build/src/environment-handlers/base.d.ts +26 -0
- package/build/src/environment-handlers/base.d.ts.map +1 -1
- package/build/src/environment-handlers/base.js +42 -1
- package/build/src/environment-handlers/node/cli/commands/db/schema/dump.d.ts +15 -0
- package/build/src/environment-handlers/node/cli/commands/db/schema/dump.d.ts.map +1 -0
- package/build/src/environment-handlers/node/cli/commands/db/schema/dump.js +35 -0
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +32 -0
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts.map +1 -0
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +117 -0
- package/build/src/environment-handlers/node.d.ts +2 -1
- package/build/src/environment-handlers/node.d.ts.map +1 -1
- package/build/src/environment-handlers/node.js +72 -4
- package/build/src/frontend-models/base.d.ts +95 -0
- package/build/src/frontend-models/base.d.ts.map +1 -0
- package/build/src/frontend-models/base.js +169 -0
- package/build/src/routes/resolver.d.ts.map +1 -1
- package/build/src/routes/resolver.js +10 -3
- package/build/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -243,6 +243,43 @@ npx velocious g:model Account
|
|
|
243
243
|
npx velocious g:model Task
|
|
244
244
|
```
|
|
245
245
|
|
|
246
|
+
## Frontend models from backend resources
|
|
247
|
+
|
|
248
|
+
You can generate lightweight frontend model classes from resource definitions in your configuration.
|
|
249
|
+
|
|
250
|
+
```js
|
|
251
|
+
export default new Configuration({
|
|
252
|
+
// ...
|
|
253
|
+
backendProjects: [
|
|
254
|
+
{
|
|
255
|
+
path: "/path/to/backend-project",
|
|
256
|
+
resources: {
|
|
257
|
+
User: {
|
|
258
|
+
attributes: ["id", "name", "email"],
|
|
259
|
+
commands: {find: "find", update: "update", destroy: "destroy"},
|
|
260
|
+
path: "/api/frontend-models/users",
|
|
261
|
+
primaryKey: "id"
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
]
|
|
266
|
+
})
|
|
267
|
+
```
|
|
268
|
+
|
|
269
|
+
Generate classes:
|
|
270
|
+
|
|
271
|
+
```bash
|
|
272
|
+
npx velocious g:frontend-models
|
|
273
|
+
```
|
|
274
|
+
|
|
275
|
+
This creates `src/frontend-models/user.js` (and one file per configured resource). Generated classes support:
|
|
276
|
+
|
|
277
|
+
- `await User.find(5)`
|
|
278
|
+
- `await user.update({...})`
|
|
279
|
+
- `await user.destroy()`
|
|
280
|
+
- Attribute methods like `user.name()` and `user.setName(...)`
|
|
281
|
+
|
|
282
|
+
|
|
246
283
|
```js
|
|
247
284
|
import Record from "velocious/build/src/database/record/index.js"
|
|
248
285
|
|
|
@@ -1159,3 +1196,54 @@ If a handed-off job does not report back within 2 hours, it is marked orphaned a
|
|
|
1159
1196
|
```bash
|
|
1160
1197
|
npx velocious server --host 0.0.0.0 --port 8082
|
|
1161
1198
|
```
|
|
1199
|
+
|
|
1200
|
+
# Authorization (CanCan-style)
|
|
1201
|
+
|
|
1202
|
+
Define resource classes with an `abilities()` method and use `can` / `cannot` rules to constrain model access.
|
|
1203
|
+
|
|
1204
|
+
```js
|
|
1205
|
+
import Ability from "velocious/build/src/authorization/ability.js"
|
|
1206
|
+
import BaseResource from "velocious/build/src/authorization/base-resource.js"
|
|
1207
|
+
import User from "@/src/models/user"
|
|
1208
|
+
|
|
1209
|
+
class UserResource extends BaseResource {
|
|
1210
|
+
static ModelClass = User
|
|
1211
|
+
|
|
1212
|
+
abilities() {
|
|
1213
|
+
const currentUser = this.currentUser()
|
|
1214
|
+
|
|
1215
|
+
if (currentUser) {
|
|
1216
|
+
this.can("read", User, {id: currentUser.id()})
|
|
1217
|
+
}
|
|
1218
|
+
}
|
|
1219
|
+
}
|
|
1220
|
+
|
|
1221
|
+
export default new Configuration({
|
|
1222
|
+
// ...
|
|
1223
|
+
abilityResolver: ({configuration, params, request, response}) => {
|
|
1224
|
+
return new Ability({
|
|
1225
|
+
context: {
|
|
1226
|
+
configuration,
|
|
1227
|
+
currentUser: undefined, // set from your auth/session layer
|
|
1228
|
+
params,
|
|
1229
|
+
request,
|
|
1230
|
+
response
|
|
1231
|
+
},
|
|
1232
|
+
resources: [UserResource]
|
|
1233
|
+
})
|
|
1234
|
+
}
|
|
1235
|
+
})
|
|
1236
|
+
```
|
|
1237
|
+
|
|
1238
|
+
Then query through authorization rules:
|
|
1239
|
+
|
|
1240
|
+
```js
|
|
1241
|
+
const users = await User.accessible().toArray()
|
|
1242
|
+
```
|
|
1243
|
+
|
|
1244
|
+
You can also pass an ability explicitly:
|
|
1245
|
+
|
|
1246
|
+
```js
|
|
1247
|
+
const ability = new Ability({context: {currentUser}, resources: [UserResource]})
|
|
1248
|
+
const users = await User.accessible(ability).toArray()
|
|
1249
|
+
```
|
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
/** @typedef {Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {ability: VelociousAuthorizationAbility, action: string, modelClass: typeof import("../database/record/index.js").default}) => void | import("../database/query/model-class-query.js").default<any>)} AbilityConditionsType */
|
|
2
|
+
/**
|
|
3
|
+
* @typedef {object} AbilityRuleType
|
|
4
|
+
* @property {string[]} actions - Actions covered by rule.
|
|
5
|
+
* @property {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
6
|
+
* @property {AbilityConditionsType | undefined} conditions - Conditions.
|
|
7
|
+
* @property {"allow" | "deny"} effect - Rule effect.
|
|
8
|
+
*/
|
|
9
|
+
/** CanCan-style ability object for query-level access control. */
|
|
10
|
+
export default class VelociousAuthorizationAbility {
|
|
11
|
+
/** @type {string[]} */
|
|
12
|
+
static CREATE: string[];
|
|
13
|
+
/** @type {string[]} */
|
|
14
|
+
static READ: string[];
|
|
15
|
+
/** @type {string[]} */
|
|
16
|
+
static UPDATE: string[];
|
|
17
|
+
/** @type {string[]} */
|
|
18
|
+
static DESTROY: string[];
|
|
19
|
+
/** @type {string[]} */
|
|
20
|
+
static CRUD: string[];
|
|
21
|
+
/**
|
|
22
|
+
* @param {object} args - Ability args.
|
|
23
|
+
* @param {Record<string, any>} [args.context] - Ability context.
|
|
24
|
+
* @param {Record<string, any>} [args.locals] - Ability locals.
|
|
25
|
+
* @param {Array<typeof BaseResource>} [args.resources] - Resource classes.
|
|
26
|
+
*/
|
|
27
|
+
constructor({ context, locals, resources }?: {
|
|
28
|
+
context?: Record<string, any>;
|
|
29
|
+
locals?: Record<string, any>;
|
|
30
|
+
resources?: Array<typeof BaseResource>;
|
|
31
|
+
});
|
|
32
|
+
context: Record<string, any>;
|
|
33
|
+
locals: Record<string, any>;
|
|
34
|
+
resources: (typeof BaseResource)[];
|
|
35
|
+
/** @type {AbilityRuleType[]} */
|
|
36
|
+
rules: AbilityRuleType[];
|
|
37
|
+
/** @type {Record<string, boolean>} */
|
|
38
|
+
loadedModelClassAbilities: Record<string, boolean>;
|
|
39
|
+
/** @returns {Record<string, any>} - Context. */
|
|
40
|
+
getContext(): Record<string, any>;
|
|
41
|
+
/** @returns {Record<string, any>} - Locals. */
|
|
42
|
+
getLocals(): Record<string, any>;
|
|
43
|
+
/** @returns {any} - Current user from context. */
|
|
44
|
+
currentUser(): any;
|
|
45
|
+
/**
|
|
46
|
+
* @param {string | string[]} actions - Action(s).
|
|
47
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
48
|
+
* @param {AbilityConditionsType} [conditions] - Conditions.
|
|
49
|
+
* @returns {void} - No return value.
|
|
50
|
+
*/
|
|
51
|
+
can(actions: string | string[], modelClass: typeof import("../database/record/index.js").default, conditions?: AbilityConditionsType): void;
|
|
52
|
+
/**
|
|
53
|
+
* @param {string | string[]} actions - Action(s).
|
|
54
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
55
|
+
* @param {AbilityConditionsType} [conditions] - Conditions.
|
|
56
|
+
* @returns {void} - No return value.
|
|
57
|
+
*/
|
|
58
|
+
cannot(actions: string | string[], modelClass: typeof import("../database/record/index.js").default, conditions?: AbilityConditionsType): void;
|
|
59
|
+
/**
|
|
60
|
+
* @param {object} args - Rule args.
|
|
61
|
+
* @param {string | string[]} args.actions - Action(s).
|
|
62
|
+
* @param {AbilityConditionsType} [args.conditions] - Conditions.
|
|
63
|
+
* @param {"allow" | "deny"} args.effect - Effect.
|
|
64
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
65
|
+
* @returns {void} - No return value.
|
|
66
|
+
*/
|
|
67
|
+
addRule({ actions, conditions, effect, modelClass }: {
|
|
68
|
+
actions: string | string[];
|
|
69
|
+
conditions?: AbilityConditionsType;
|
|
70
|
+
effect: "allow" | "deny";
|
|
71
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
72
|
+
}): void;
|
|
73
|
+
/**
|
|
74
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
75
|
+
* @returns {void} - No return value.
|
|
76
|
+
*/
|
|
77
|
+
loadAbilitiesForModelClass(modelClass: typeof import("../database/record/index.js").default): void;
|
|
78
|
+
/**
|
|
79
|
+
* @param {object} args - Query args.
|
|
80
|
+
* @param {string} args.action - Requested action.
|
|
81
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
82
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
83
|
+
* @returns {import("../database/query/model-class-query.js").default<any>} - Authorized query.
|
|
84
|
+
*/
|
|
85
|
+
applyToQuery({ action, modelClass, query }: {
|
|
86
|
+
action: string;
|
|
87
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
88
|
+
query: import("../database/query/model-class-query.js").default<any>;
|
|
89
|
+
}): import("../database/query/model-class-query.js").default<any>;
|
|
90
|
+
/**
|
|
91
|
+
* @param {object} args - Rule lookup args.
|
|
92
|
+
* @param {string} args.action - Action.
|
|
93
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
94
|
+
* @returns {AbilityRuleType[]} - Matching rules.
|
|
95
|
+
*/
|
|
96
|
+
rulesFor({ action, modelClass }: {
|
|
97
|
+
action: string;
|
|
98
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
99
|
+
}): AbilityRuleType[];
|
|
100
|
+
/**
|
|
101
|
+
* @param {object} args - SQL args.
|
|
102
|
+
* @param {string} args.action - Action.
|
|
103
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
104
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Base query.
|
|
105
|
+
* @param {AbilityRuleType[]} args.rules - Rules.
|
|
106
|
+
* @returns {string[]} - SQL condition parts.
|
|
107
|
+
*/
|
|
108
|
+
conditionSqlParts({ action, modelClass, query, rules }: {
|
|
109
|
+
action: string;
|
|
110
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
111
|
+
query: import("../database/query/model-class-query.js").default<any>;
|
|
112
|
+
rules: AbilityRuleType[];
|
|
113
|
+
}): string[];
|
|
114
|
+
/**
|
|
115
|
+
* @param {object} args - Deny args.
|
|
116
|
+
* @param {string} args.action - Action.
|
|
117
|
+
* @param {AbilityRuleType[]} args.denyRules - Deny rules.
|
|
118
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
119
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
120
|
+
* @returns {void} - No return value.
|
|
121
|
+
*/
|
|
122
|
+
applyDenyRules({ action, denyRules, modelClass, query }: {
|
|
123
|
+
action: string;
|
|
124
|
+
denyRules: AbilityRuleType[];
|
|
125
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
126
|
+
query: import("../database/query/model-class-query.js").default<any>;
|
|
127
|
+
}): void;
|
|
128
|
+
/**
|
|
129
|
+
* @param {object} args - Condition args.
|
|
130
|
+
* @param {string} args.action - Action.
|
|
131
|
+
* @param {AbilityConditionsType} args.conditions - Rule conditions.
|
|
132
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
133
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
134
|
+
* @returns {void | import("../database/query/model-class-query.js").default<any>} - Optional replacement query.
|
|
135
|
+
*/
|
|
136
|
+
applyRuleCondition({ action, conditions, modelClass, query }: {
|
|
137
|
+
action: string;
|
|
138
|
+
conditions: AbilityConditionsType;
|
|
139
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
140
|
+
query: import("../database/query/model-class-query.js").default<any>;
|
|
141
|
+
}): void | import("../database/query/model-class-query.js").default<any>;
|
|
142
|
+
}
|
|
143
|
+
export type AbilityConditionsType = Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {
|
|
144
|
+
ability: VelociousAuthorizationAbility;
|
|
145
|
+
action: string;
|
|
146
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
147
|
+
}) => void | import("../database/query/model-class-query.js").default<any>);
|
|
148
|
+
export type AbilityRuleType = {
|
|
149
|
+
/**
|
|
150
|
+
* - Actions covered by rule.
|
|
151
|
+
*/
|
|
152
|
+
actions: string[];
|
|
153
|
+
/**
|
|
154
|
+
* - Model class.
|
|
155
|
+
*/
|
|
156
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
157
|
+
/**
|
|
158
|
+
* - Conditions.
|
|
159
|
+
*/
|
|
160
|
+
conditions: AbilityConditionsType | undefined;
|
|
161
|
+
/**
|
|
162
|
+
* - Rule effect.
|
|
163
|
+
*/
|
|
164
|
+
effect: "allow" | "deny";
|
|
165
|
+
};
|
|
166
|
+
import BaseResource from "./base-resource.js";
|
|
167
|
+
//# sourceMappingURL=ability.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ability.d.ts","sourceRoot":"","sources":["../../../src/authorization/ability.js"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import BaseResource from "./base-resource.js";
|
|
3
|
+
/** @typedef {Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {ability: VelociousAuthorizationAbility, action: string, modelClass: typeof import("../database/record/index.js").default}) => void | import("../database/query/model-class-query.js").default<any>)} AbilityConditionsType */
|
|
4
|
+
/**
|
|
5
|
+
* @typedef {object} AbilityRuleType
|
|
6
|
+
* @property {string[]} actions - Actions covered by rule.
|
|
7
|
+
* @property {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
8
|
+
* @property {AbilityConditionsType | undefined} conditions - Conditions.
|
|
9
|
+
* @property {"allow" | "deny"} effect - Rule effect.
|
|
10
|
+
*/
|
|
11
|
+
/** CanCan-style ability object for query-level access control. */
|
|
12
|
+
export default class VelociousAuthorizationAbility {
|
|
13
|
+
/** @type {string[]} */
|
|
14
|
+
static CREATE = ["create"];
|
|
15
|
+
/** @type {string[]} */
|
|
16
|
+
static READ = ["read"];
|
|
17
|
+
/** @type {string[]} */
|
|
18
|
+
static UPDATE = ["update"];
|
|
19
|
+
/** @type {string[]} */
|
|
20
|
+
static DESTROY = ["destroy"];
|
|
21
|
+
/** @type {string[]} */
|
|
22
|
+
static CRUD = ["create", "read", "update", "destroy"];
|
|
23
|
+
/**
|
|
24
|
+
* @param {object} args - Ability args.
|
|
25
|
+
* @param {Record<string, any>} [args.context] - Ability context.
|
|
26
|
+
* @param {Record<string, any>} [args.locals] - Ability locals.
|
|
27
|
+
* @param {Array<typeof BaseResource>} [args.resources] - Resource classes.
|
|
28
|
+
*/
|
|
29
|
+
constructor({ context = {}, locals = {}, resources = [] } = {}) {
|
|
30
|
+
this.context = context;
|
|
31
|
+
this.locals = locals;
|
|
32
|
+
this.resources = resources;
|
|
33
|
+
/** @type {AbilityRuleType[]} */
|
|
34
|
+
this.rules = [];
|
|
35
|
+
/** @type {Record<string, boolean>} */
|
|
36
|
+
this.loadedModelClassAbilities = {};
|
|
37
|
+
}
|
|
38
|
+
/** @returns {Record<string, any>} - Context. */
|
|
39
|
+
getContext() {
|
|
40
|
+
return this.context;
|
|
41
|
+
}
|
|
42
|
+
/** @returns {Record<string, any>} - Locals. */
|
|
43
|
+
getLocals() {
|
|
44
|
+
return this.locals;
|
|
45
|
+
}
|
|
46
|
+
/** @returns {any} - Current user from context. */
|
|
47
|
+
currentUser() {
|
|
48
|
+
return this.context.currentUser;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* @param {string | string[]} actions - Action(s).
|
|
52
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
53
|
+
* @param {AbilityConditionsType} [conditions] - Conditions.
|
|
54
|
+
* @returns {void} - No return value.
|
|
55
|
+
*/
|
|
56
|
+
can(actions, modelClass, conditions) {
|
|
57
|
+
this.addRule({ actions, conditions, effect: "allow", modelClass });
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* @param {string | string[]} actions - Action(s).
|
|
61
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
62
|
+
* @param {AbilityConditionsType} [conditions] - Conditions.
|
|
63
|
+
* @returns {void} - No return value.
|
|
64
|
+
*/
|
|
65
|
+
cannot(actions, modelClass, conditions) {
|
|
66
|
+
this.addRule({ actions, conditions, effect: "deny", modelClass });
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* @param {object} args - Rule args.
|
|
70
|
+
* @param {string | string[]} args.actions - Action(s).
|
|
71
|
+
* @param {AbilityConditionsType} [args.conditions] - Conditions.
|
|
72
|
+
* @param {"allow" | "deny"} args.effect - Effect.
|
|
73
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
74
|
+
* @returns {void} - No return value.
|
|
75
|
+
*/
|
|
76
|
+
addRule({ actions, conditions, effect, modelClass }) {
|
|
77
|
+
const normalizedActions = Array.isArray(actions) ? actions : [actions];
|
|
78
|
+
this.rules.push({ actions: normalizedActions, conditions, effect, modelClass });
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
82
|
+
* @returns {void} - No return value.
|
|
83
|
+
*/
|
|
84
|
+
loadAbilitiesForModelClass(modelClass) {
|
|
85
|
+
const key = modelClass.name;
|
|
86
|
+
if (this.loadedModelClassAbilities[key])
|
|
87
|
+
return;
|
|
88
|
+
this.loadedModelClassAbilities[key] = true;
|
|
89
|
+
for (const ResourceClass of this.resources) {
|
|
90
|
+
const resourceModelClass = ResourceClass.modelClass();
|
|
91
|
+
if (!resourceModelClass)
|
|
92
|
+
continue;
|
|
93
|
+
if (resourceModelClass !== modelClass)
|
|
94
|
+
continue;
|
|
95
|
+
const resourceInstance = new ResourceClass({
|
|
96
|
+
ability: this,
|
|
97
|
+
context: this.context,
|
|
98
|
+
locals: this.locals
|
|
99
|
+
});
|
|
100
|
+
resourceInstance.abilities();
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* @param {object} args - Query args.
|
|
105
|
+
* @param {string} args.action - Requested action.
|
|
106
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
107
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
108
|
+
* @returns {import("../database/query/model-class-query.js").default<any>} - Authorized query.
|
|
109
|
+
*/
|
|
110
|
+
applyToQuery({ action, modelClass, query }) {
|
|
111
|
+
this.loadAbilitiesForModelClass(modelClass);
|
|
112
|
+
const applicableRules = this.rulesFor({ action, modelClass });
|
|
113
|
+
const allowRules = applicableRules.filter((rule) => rule.effect === "allow");
|
|
114
|
+
const denyRules = applicableRules.filter((rule) => rule.effect === "deny");
|
|
115
|
+
if (allowRules.length === 0) {
|
|
116
|
+
return query.where("1=0");
|
|
117
|
+
}
|
|
118
|
+
if (allowRules.some((rule) => !rule.conditions)) {
|
|
119
|
+
this.applyDenyRules({ action, denyRules, modelClass, query });
|
|
120
|
+
return query;
|
|
121
|
+
}
|
|
122
|
+
const allowSqlParts = this.conditionSqlParts({ action, modelClass, query, rules: allowRules });
|
|
123
|
+
if (allowSqlParts.length === 0) {
|
|
124
|
+
return query.where("1=0");
|
|
125
|
+
}
|
|
126
|
+
query.where(`(${allowSqlParts.join(" OR ")})`);
|
|
127
|
+
this.applyDenyRules({ action, denyRules, modelClass, query });
|
|
128
|
+
return query;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* @param {object} args - Rule lookup args.
|
|
132
|
+
* @param {string} args.action - Action.
|
|
133
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
134
|
+
* @returns {AbilityRuleType[]} - Matching rules.
|
|
135
|
+
*/
|
|
136
|
+
rulesFor({ action, modelClass }) {
|
|
137
|
+
return this.rules.filter((rule) => {
|
|
138
|
+
if (rule.modelClass !== modelClass)
|
|
139
|
+
return false;
|
|
140
|
+
return rule.actions.includes(action) || rule.actions.includes("manage");
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* @param {object} args - SQL args.
|
|
145
|
+
* @param {string} args.action - Action.
|
|
146
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
147
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Base query.
|
|
148
|
+
* @param {AbilityRuleType[]} args.rules - Rules.
|
|
149
|
+
* @returns {string[]} - SQL condition parts.
|
|
150
|
+
*/
|
|
151
|
+
conditionSqlParts({ action, modelClass, query, rules }) {
|
|
152
|
+
const pk = modelClass.primaryKey();
|
|
153
|
+
const quotedBaseTable = query.driver.quoteTable(modelClass.tableName());
|
|
154
|
+
const quotedPk = query.driver.quoteColumn(pk);
|
|
155
|
+
const sqlParts = [];
|
|
156
|
+
for (const rule of rules) {
|
|
157
|
+
if (!rule.conditions)
|
|
158
|
+
continue;
|
|
159
|
+
const scopedQuery = modelClass._newQuery();
|
|
160
|
+
const resultQuery = this.applyRuleCondition({
|
|
161
|
+
action,
|
|
162
|
+
conditions: rule.conditions,
|
|
163
|
+
modelClass,
|
|
164
|
+
query: scopedQuery
|
|
165
|
+
});
|
|
166
|
+
const finalQuery = resultQuery || scopedQuery;
|
|
167
|
+
const selectedPkSql = `${quotedBaseTable}.${quotedPk}`;
|
|
168
|
+
finalQuery.select(selectedPkSql);
|
|
169
|
+
sqlParts.push(`${quotedBaseTable}.${quotedPk} IN (${finalQuery.toSql()})`);
|
|
170
|
+
}
|
|
171
|
+
return sqlParts;
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* @param {object} args - Deny args.
|
|
175
|
+
* @param {string} args.action - Action.
|
|
176
|
+
* @param {AbilityRuleType[]} args.denyRules - Deny rules.
|
|
177
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
178
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
179
|
+
* @returns {void} - No return value.
|
|
180
|
+
*/
|
|
181
|
+
applyDenyRules({ action, denyRules, modelClass, query }) {
|
|
182
|
+
if (denyRules.length === 0)
|
|
183
|
+
return;
|
|
184
|
+
if (denyRules.some((rule) => !rule.conditions)) {
|
|
185
|
+
query.where("1=0");
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
const denySqlParts = this.conditionSqlParts({ action, modelClass, query, rules: denyRules });
|
|
189
|
+
if (denySqlParts.length > 0) {
|
|
190
|
+
query.where(`NOT (${denySqlParts.join(" OR ")})`);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* @param {object} args - Condition args.
|
|
195
|
+
* @param {string} args.action - Action.
|
|
196
|
+
* @param {AbilityConditionsType} args.conditions - Rule conditions.
|
|
197
|
+
* @param {typeof import("../database/record/index.js").default} args.modelClass - Model class.
|
|
198
|
+
* @param {import("../database/query/model-class-query.js").default<any>} args.query - Query.
|
|
199
|
+
* @returns {void | import("../database/query/model-class-query.js").default<any>} - Optional replacement query.
|
|
200
|
+
*/
|
|
201
|
+
applyRuleCondition({ action, conditions, modelClass, query }) {
|
|
202
|
+
if (typeof conditions === "string") {
|
|
203
|
+
query.where(conditions);
|
|
204
|
+
return;
|
|
205
|
+
}
|
|
206
|
+
if (typeof conditions === "function") {
|
|
207
|
+
return conditions(query, {
|
|
208
|
+
ability: this,
|
|
209
|
+
action,
|
|
210
|
+
modelClass
|
|
211
|
+
});
|
|
212
|
+
}
|
|
213
|
+
query.where(conditions);
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ability.js","sourceRoot":"","sources":["../../../src/authorization/ability.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport BaseResource from \"./base-resource.js\"\n\n/** @typedef {Record<string, any> | string | ((query: import(\"../database/query/model-class-query.js\").default<any>, args: {ability: VelociousAuthorizationAbility, action: string, modelClass: typeof import(\"../database/record/index.js\").default}) => void | import(\"../database/query/model-class-query.js\").default<any>)} AbilityConditionsType */\n\n/**\n * @typedef {object} AbilityRuleType\n * @property {string[]} actions - Actions covered by rule.\n * @property {typeof import(\"../database/record/index.js\").default} modelClass - Model class.\n * @property {AbilityConditionsType | undefined} conditions - Conditions.\n * @property {\"allow\" | \"deny\"} effect - Rule effect.\n */\n\n/** CanCan-style ability object for query-level access control. */\nexport default class VelociousAuthorizationAbility {\n  /** @type {string[]} */\n  static CREATE = [\"create\"]\n\n  /** @type {string[]} */\n  static READ = [\"read\"]\n\n  /** @type {string[]} */\n  static UPDATE = [\"update\"]\n\n  /** @type {string[]} */\n  static DESTROY = [\"destroy\"]\n\n  /** @type {string[]} */\n  static CRUD = [\"create\", \"read\", \"update\", \"destroy\"]\n\n  /**\n   * @param {object} args - Ability args.\n   * @param {Record<string, any>} [args.context] - Ability context.\n   * @param {Record<string, any>} [args.locals] - Ability locals.\n   * @param {Array<typeof BaseResource>} [args.resources] - Resource classes.\n   */\n  constructor({context = {}, locals = {}, resources = []} = {}) {\n    this.context = context\n    this.locals = locals\n    this.resources = resources\n\n    /** @type {AbilityRuleType[]} */\n    this.rules = []\n\n    /** @type {Record<string, boolean>} */\n    this.loadedModelClassAbilities = {}\n  }\n\n  /** @returns {Record<string, any>} - Context. */\n  getContext() {\n    return this.context\n  }\n\n  /** @returns {Record<string, any>} - Locals. */\n  getLocals() {\n    return this.locals\n  }\n\n  /** @returns {any} - Current user from context. */\n  currentUser() {\n    return this.context.currentUser\n  }\n\n  /**\n   * @param {string | string[]} actions - Action(s).\n   * @param {typeof import(\"../database/record/index.js\").default} modelClass - Model class.\n   * @param {AbilityConditionsType} [conditions] - Conditions.\n   * @returns {void} - No return value.\n   */\n  can(actions, modelClass, conditions) {\n    this.addRule({actions, conditions, effect: \"allow\", modelClass})\n  }\n\n  /**\n   * @param {string | string[]} actions - Action(s).\n   * @param {typeof import(\"../database/record/index.js\").default} modelClass - Model class.\n   * @param {AbilityConditionsType} [conditions] - Conditions.\n   * @returns {void} - No return value.\n   */\n  cannot(actions, modelClass, conditions) {\n    this.addRule({actions, conditions, effect: \"deny\", modelClass})\n  }\n\n  /**\n   * @param {object} args - Rule args.\n   * @param {string | string[]} args.actions - Action(s).\n   * @param {AbilityConditionsType} [args.conditions] - Conditions.\n   * @param {\"allow\" | \"deny\"} args.effect - Effect.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @returns {void} - No return value.\n   */\n  addRule({actions, conditions, effect, modelClass}) {\n    const normalizedActions = Array.isArray(actions) ? actions : [actions]\n\n    this.rules.push({actions: normalizedActions, conditions, effect, modelClass})\n  }\n\n  /**\n   * @param {typeof import(\"../database/record/index.js\").default} modelClass - Model class.\n   * @returns {void} - No return value.\n   */\n  loadAbilitiesForModelClass(modelClass) {\n    const key = modelClass.name\n\n    if (this.loadedModelClassAbilities[key]) return\n\n    this.loadedModelClassAbilities[key] = true\n\n    for (const ResourceClass of this.resources) {\n      const resourceModelClass = ResourceClass.modelClass()\n\n      if (!resourceModelClass) continue\n      if (resourceModelClass !== modelClass) continue\n\n      const resourceInstance = new ResourceClass({\n        ability: this,\n        context: this.context,\n        locals: this.locals\n      })\n\n      resourceInstance.abilities()\n    }\n  }\n\n  /**\n   * @param {object} args - Query args.\n   * @param {string} args.action - Requested action.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @param {import(\"../database/query/model-class-query.js\").default<any>} args.query - Query.\n   * @returns {import(\"../database/query/model-class-query.js\").default<any>} - Authorized query.\n   */\n  applyToQuery({action, modelClass, query}) {\n    this.loadAbilitiesForModelClass(modelClass)\n\n    const applicableRules = this.rulesFor({action, modelClass})\n    const allowRules = applicableRules.filter((rule) => rule.effect === \"allow\")\n    const denyRules = applicableRules.filter((rule) => rule.effect === \"deny\")\n\n    if (allowRules.length === 0) {\n      return query.where(\"1=0\")\n    }\n\n    if (allowRules.some((rule) => !rule.conditions)) {\n      this.applyDenyRules({action, denyRules, modelClass, query})\n      return query\n    }\n\n    const allowSqlParts = this.conditionSqlParts({action, modelClass, query, rules: allowRules})\n\n    if (allowSqlParts.length === 0) {\n      return query.where(\"1=0\")\n    }\n\n    query.where(`(${allowSqlParts.join(\" OR \")})`)\n    this.applyDenyRules({action, denyRules, modelClass, query})\n\n    return query\n  }\n\n  /**\n   * @param {object} args - Rule lookup args.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @returns {AbilityRuleType[]} - Matching rules.\n   */\n  rulesFor({action, modelClass}) {\n    return this.rules.filter((rule) => {\n      if (rule.modelClass !== modelClass) return false\n\n      return rule.actions.includes(action) || rule.actions.includes(\"manage\")\n    })\n  }\n\n  /**\n   * @param {object} args - SQL args.\n   * @param {string} args.action - Action.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @param {import(\"../database/query/model-class-query.js\").default<any>} args.query - Base query.\n   * @param {AbilityRuleType[]} args.rules - Rules.\n   * @returns {string[]} - SQL condition parts.\n   */\n  conditionSqlParts({action, modelClass, query, rules}) {\n    const pk = modelClass.primaryKey()\n    const quotedBaseTable = query.driver.quoteTable(modelClass.tableName())\n    const quotedPk = query.driver.quoteColumn(pk)\n    const sqlParts = []\n\n    for (const rule of rules) {\n      if (!rule.conditions) continue\n\n      const scopedQuery = modelClass._newQuery()\n      const resultQuery = this.applyRuleCondition({\n        action,\n        conditions: rule.conditions,\n        modelClass,\n        query: scopedQuery\n      })\n      const finalQuery = resultQuery || scopedQuery\n      const selectedPkSql = `${quotedBaseTable}.${quotedPk}`\n\n      finalQuery.select(selectedPkSql)\n\n      sqlParts.push(`${quotedBaseTable}.${quotedPk} IN (${finalQuery.toSql()})`)\n    }\n\n    return sqlParts\n  }\n\n  /**\n   * @param {object} args - Deny args.\n   * @param {string} args.action - Action.\n   * @param {AbilityRuleType[]} args.denyRules - Deny rules.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @param {import(\"../database/query/model-class-query.js\").default<any>} args.query - Query.\n   * @returns {void} - No return value.\n   */\n  applyDenyRules({action, denyRules, modelClass, query}) {\n    if (denyRules.length === 0) return\n\n    if (denyRules.some((rule) => !rule.conditions)) {\n      query.where(\"1=0\")\n      return\n    }\n\n    const denySqlParts = this.conditionSqlParts({action, modelClass, query, rules: denyRules})\n\n    if (denySqlParts.length > 0) {\n      query.where(`NOT (${denySqlParts.join(\" OR \")})`)\n    }\n  }\n\n  /**\n   * @param {object} args - Condition args.\n   * @param {string} args.action - Action.\n   * @param {AbilityConditionsType} args.conditions - Rule conditions.\n   * @param {typeof import(\"../database/record/index.js\").default} args.modelClass - Model class.\n   * @param {import(\"../database/query/model-class-query.js\").default<any>} args.query - Query.\n   * @returns {void | import(\"../database/query/model-class-query.js\").default<any>} - Optional replacement query.\n   */\n  applyRuleCondition({action, conditions, modelClass, query}) {\n    if (typeof conditions === \"string\") {\n      query.where(conditions)\n      return\n    }\n\n    if (typeof conditions === \"function\") {\n      return conditions(query, {\n        ability: this,\n        action,\n        modelClass\n      })\n    }\n\n    query.where(conditions)\n  }\n}\n"]}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
/** Base class for authorization resources defining abilities for a model. */
|
|
2
|
+
export default class AuthorizationBaseResource {
|
|
3
|
+
/** @type {typeof import("../database/record/index.js").default | undefined} */
|
|
4
|
+
static ModelClass: typeof import("../database/record/index.js").default | undefined;
|
|
5
|
+
/**
|
|
6
|
+
* @returns {typeof import("../database/record/index.js").default | undefined} - Model class handled by this resource.
|
|
7
|
+
*/
|
|
8
|
+
static modelClass(): typeof import("../database/record/index.js").default | undefined;
|
|
9
|
+
/**
|
|
10
|
+
* @param {object} args - Resource args.
|
|
11
|
+
* @param {import("./ability.js").default} args.ability - Ability instance.
|
|
12
|
+
* @param {Record<string, any>} [args.context] - Ability context.
|
|
13
|
+
* @param {Record<string, any>} [args.locals] - Ability locals.
|
|
14
|
+
*/
|
|
15
|
+
constructor({ ability, context, locals }: {
|
|
16
|
+
ability: import("./ability.js").default;
|
|
17
|
+
context?: Record<string, any>;
|
|
18
|
+
locals?: Record<string, any>;
|
|
19
|
+
});
|
|
20
|
+
ability: import("./ability.js").default;
|
|
21
|
+
context: Record<string, any>;
|
|
22
|
+
locals: Record<string, any>;
|
|
23
|
+
/**
|
|
24
|
+
* @param {string | string[]} actions - Ability action(s).
|
|
25
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
26
|
+
* @param {Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {ability: import("./ability.js").default, action: string, modelClass: typeof import("../database/record/index.js").default}) => void | import("../database/query/model-class-query.js").default<any>)} [conditions] - Conditions.
|
|
27
|
+
* @returns {void} - No return value.
|
|
28
|
+
*/
|
|
29
|
+
can(actions: string | string[], modelClass: typeof import("../database/record/index.js").default, conditions?: Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {
|
|
30
|
+
ability: import("./ability.js").default;
|
|
31
|
+
action: string;
|
|
32
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
33
|
+
}) => void | import("../database/query/model-class-query.js").default<any>)): void;
|
|
34
|
+
/**
|
|
35
|
+
* @param {string | string[]} actions - Ability action(s).
|
|
36
|
+
* @param {typeof import("../database/record/index.js").default} modelClass - Model class.
|
|
37
|
+
* @param {Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {ability: import("./ability.js").default, action: string, modelClass: typeof import("../database/record/index.js").default}) => void | import("../database/query/model-class-query.js").default<any>)} [conditions] - Conditions.
|
|
38
|
+
* @returns {void} - No return value.
|
|
39
|
+
*/
|
|
40
|
+
cannot(actions: string | string[], modelClass: typeof import("../database/record/index.js").default, conditions?: Record<string, any> | string | ((query: import("../database/query/model-class-query.js").default<any>, args: {
|
|
41
|
+
ability: import("./ability.js").default;
|
|
42
|
+
action: string;
|
|
43
|
+
modelClass: typeof import("../database/record/index.js").default;
|
|
44
|
+
}) => void | import("../database/query/model-class-query.js").default<any>)): void;
|
|
45
|
+
/** @returns {Record<string, any>} - Ability context. */
|
|
46
|
+
getContext(): Record<string, any>;
|
|
47
|
+
/** @returns {Record<string, any>} - Ability locals. */
|
|
48
|
+
getLocals(): Record<string, any>;
|
|
49
|
+
/** @returns {any} - Current user from context. */
|
|
50
|
+
currentUser(): any;
|
|
51
|
+
/** @returns {import("../http-server/client/request.js").default | import("../http-server/client/websocket-request.js").default | undefined} - Request from context. */
|
|
52
|
+
request(): import("../http-server/client/request.js").default | import("../http-server/client/websocket-request.js").default | undefined;
|
|
53
|
+
/** @returns {Record<string, any> | undefined} - Params from context. */
|
|
54
|
+
params(): Record<string, any> | undefined;
|
|
55
|
+
/** @returns {void} - Implement in subclasses to define abilities. */
|
|
56
|
+
abilities(): void;
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=base-resource.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"base-resource.d.ts","sourceRoot":"","sources":["../../../src/authorization/base-resource.js"],"names":[],"mappings":"AAEA,6EAA6E;AAC7E;IACE,+EAA+E;IAC/E,mBADW,cAAc,6BAA6B,EAAE,OAAO,GAAG,SAAS,CAC9C;IAc7B;;OAEG;IACH,qBAFa,cAAc,6BAA6B,EAAE,OAAO,GAAG,SAAS,CAI5E;IAjBD;;;;;OAKG;IACH,0CAJG;QAA6C,OAAO,EAA5C,OAAO,cAAc,EAAE,OAAO;QACH,OAAO,GAAlC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;QACQ,MAAM,GAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC7B,EAKA;IAHC,wCAAsB;IACtB,6BAAsB;IACtB,4BAAoB;IAUtB;;;;;OAKG;IACH,aALW,MAAM,GAAG,MAAM,EAAE,cACjB,cAAc,6BAA6B,EAAE,OAAO,eACpD,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,MAAM,GAAG,CAAC,CAAC,KAAK,EAAE,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE;QAAC,OAAO,EAAE,OAAO,cAAc,EAAE,OAAO,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,cAAc,6BAA6B,EAAE,OAAO,CAAA;KAAC,KAAK,IAAI,GAAG,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,GAChT,IAAI,CAIhB;IAED;;;;;OAKG;IACH,gBALW,MAAM,GAAG,MAAM,EAAE,cACjB,cAAc,6BAA6B,EAAE,OAAO,eACpD,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,MAAM,GAAG,CAAC,CAAC,KAAK,EAAE,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE;QAAC,OAAO,EAAE,OAAO,cAAc,EAAE,OAAO,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,UAAU,EAAE,cAAc,6BAA6B,EAAE,OAAO,CAAA;KAAC,KAAK,IAAI,GAAG,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,GAChT,IAAI,CAIhB;IAED,wDAAwD;IACxD,cADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAGhC;IAED,uDAAuD;IACvD,aADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAGhC;IAED,kDAAkD;IAClD,eADc,GAAG,CAGhB;IAED,uKAAuK;IACvK,WADc,OAAO,kCAAkC,EAAE,OAAO,GAAG,OAAO,4CAA4C,EAAE,OAAO,GAAG,SAAS,CAG1I;IAED,wEAAwE;IACxE,UADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,SAAS,CAG5C;IAED,qEAAqE;IACrE,aADc,IAAI,CAGjB;CACF"}
|