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.
Files changed (44) hide show
  1. package/README.md +88 -0
  2. package/build/src/authorization/ability.d.ts +167 -0
  3. package/build/src/authorization/ability.d.ts.map +1 -0
  4. package/build/src/authorization/ability.js +216 -0
  5. package/build/src/authorization/base-resource.d.ts +58 -0
  6. package/build/src/authorization/base-resource.d.ts.map +1 -0
  7. package/build/src/authorization/base-resource.js +66 -0
  8. package/build/src/cli/commands/db/schema/dump.d.ts +2 -10
  9. package/build/src/cli/commands/db/schema/dump.d.ts.map +1 -1
  10. package/build/src/cli/commands/db/schema/dump.js +3 -29
  11. package/build/src/cli/commands/generate/frontend-models.d.ts +7 -0
  12. package/build/src/cli/commands/generate/frontend-models.d.ts.map +1 -0
  13. package/build/src/cli/commands/generate/frontend-models.js +9 -0
  14. package/build/src/configuration-types.d.ts +73 -0
  15. package/build/src/configuration-types.d.ts.map +1 -1
  16. package/build/src/configuration-types.js +23 -1
  17. package/build/src/configuration.d.ts +45 -2
  18. package/build/src/configuration.d.ts.map +1 -1
  19. package/build/src/configuration.js +60 -2
  20. package/build/src/controller.d.ts +2 -0
  21. package/build/src/controller.d.ts.map +1 -1
  22. package/build/src/controller.js +5 -1
  23. package/build/src/database/record/index.d.ts +14 -0
  24. package/build/src/database/record/index.d.ts.map +1 -1
  25. package/build/src/database/record/index.js +28 -1
  26. package/build/src/environment-handlers/base.d.ts +26 -0
  27. package/build/src/environment-handlers/base.d.ts.map +1 -1
  28. package/build/src/environment-handlers/base.js +42 -1
  29. package/build/src/environment-handlers/node/cli/commands/db/schema/dump.d.ts +15 -0
  30. package/build/src/environment-handlers/node/cli/commands/db/schema/dump.d.ts.map +1 -0
  31. package/build/src/environment-handlers/node/cli/commands/db/schema/dump.js +35 -0
  32. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +32 -0
  33. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts.map +1 -0
  34. package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +117 -0
  35. package/build/src/environment-handlers/node.d.ts +2 -1
  36. package/build/src/environment-handlers/node.d.ts.map +1 -1
  37. package/build/src/environment-handlers/node.js +72 -4
  38. package/build/src/frontend-models/base.d.ts +95 -0
  39. package/build/src/frontend-models/base.d.ts.map +1 -0
  40. package/build/src/frontend-models/base.js +169 -0
  41. package/build/src/routes/resolver.d.ts.map +1 -1
  42. package/build/src/routes/resolver.js +10 -3
  43. package/build/tsconfig.tsbuildinfo +1 -1
  44. 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"}