@ditojs/server 0.275.0 → 1.0.0
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/package.json +25 -43
- package/src/app/Application.js +105 -111
- package/src/app/Validator.js +6 -3
- package/src/app/index.js +2 -2
- package/src/cli/db/createMigration.js +1 -1
- package/src/cli/db/index.js +7 -7
- package/src/cli/db/listAssetConfig.js +1 -1
- package/src/cli/db/reset.js +1 -1
- package/src/cli/index.js +14 -5
- package/src/controllers/AdminController.js +181 -158
- package/src/controllers/CollectionController.js +8 -29
- package/src/controllers/Controller.js +71 -76
- package/src/controllers/ControllerAction.js +39 -17
- package/src/controllers/MemberAction.js +2 -2
- package/src/controllers/ModelController.js +4 -4
- package/src/controllers/RelationController.js +3 -3
- package/src/controllers/{UserController.js → UsersController.js} +8 -13
- package/src/controllers/index.js +5 -5
- package/src/decorators/action.js +3 -3
- package/src/decorators/authorize.js +1 -1
- package/src/decorators/index.js +6 -6
- package/src/decorators/parameters.js +1 -1
- package/src/decorators/returns.js +1 -1
- package/src/decorators/scope.js +1 -1
- package/src/decorators/transacted.js +1 -1
- package/src/errors/AssetError.js +1 -1
- package/src/errors/AuthenticationError.js +1 -1
- package/src/errors/AuthorizationError.js +1 -1
- package/src/errors/ControllerError.js +1 -1
- package/src/errors/DatabaseError.js +12 -3
- package/src/errors/GraphError.js +1 -1
- package/src/errors/ModelError.js +1 -1
- package/src/errors/NotFoundError.js +1 -1
- package/src/errors/NotImplementedError.js +1 -1
- package/src/errors/QueryBuilderError.js +1 -1
- package/src/errors/RelationError.js +1 -1
- package/src/errors/ValidationError.js +1 -1
- package/src/errors/WrappedError.js +1 -1
- package/src/errors/index.js +14 -14
- package/src/graph/DitoGraphProcessor.js +2 -1
- package/src/graph/graph.js +1 -1
- package/src/graph/index.js +3 -3
- package/src/index.js +11 -9
- package/src/lib/index.js +2 -2
- package/src/middleware/createTransaction.js +1 -1
- package/src/middleware/findRoute.js +3 -2
- package/src/middleware/handleConnectMiddleware.js +88 -0
- package/src/middleware/handleError.js +1 -1
- package/src/middleware/handleRoute.js +3 -3
- package/src/middleware/index.js +8 -7
- package/src/mixins/AssetMixin.js +1 -1
- package/src/mixins/UserMixin.js +1 -1
- package/src/mixins/index.js +4 -4
- package/src/models/AssetModel.js +2 -2
- package/src/models/Model.js +16 -12
- package/src/models/RelationAccessor.js +1 -1
- package/src/models/SessionModel.js +2 -2
- package/src/models/TimeStampedModel.js +2 -2
- package/src/models/UserModel.js +2 -2
- package/src/models/definitions/assets.js +1 -1
- package/src/models/definitions/filters.js +57 -44
- package/src/models/definitions/hooks.js +1 -1
- package/src/models/definitions/index.js +9 -9
- package/src/models/definitions/modifiers.js +1 -1
- package/src/models/definitions/options.js +1 -1
- package/src/models/definitions/properties.js +2 -2
- package/src/models/definitions/relations.js +1 -1
- package/src/models/definitions/schema.js +1 -1
- package/src/models/definitions/scopes.js +2 -2
- package/src/models/index.js +5 -5
- package/src/query/QueryBuilder.js +5 -5
- package/src/query/QueryFilters.js +50 -50
- package/src/query/QueryParameters.js +2 -2
- package/src/query/index.js +3 -3
- package/src/schema/formats/index.js +2 -2
- package/src/schema/index.js +4 -4
- package/src/schema/keywords/_relate.js +1 -1
- package/src/schema/keywords/index.js +12 -12
- package/src/schema/properties.test.js +1 -1
- package/src/schema/relations.js +1 -1
- package/src/schema/relations.test.js +2 -2
- package/src/services/index.js +1 -1
- package/src/storage/DiskStorage.js +1 -1
- package/src/storage/S3Storage.js +4 -4
- package/src/storage/Storage.js +1 -1
- package/src/storage/index.js +4 -4
- package/src/utils/function.test.js +1 -1
- package/src/utils/handler.js +17 -0
- package/src/utils/index.js +8 -7
- package/src/utils/object.test.js +1 -1
- package/lib/app/Application.js +0 -961
- package/lib/app/SessionStore.js +0 -40
- package/lib/app/Validator.js +0 -355
- package/lib/app/index.js +0 -26
- package/lib/cli/console.js +0 -175
- package/lib/cli/db/createMigration.js +0 -237
- package/lib/cli/db/index.js +0 -66
- package/lib/cli/db/listAssetConfig.js +0 -16
- package/lib/cli/db/migrate.js +0 -15
- package/lib/cli/db/reset.js +0 -27
- package/lib/cli/db/rollback.js +0 -15
- package/lib/cli/db/seed.js +0 -104
- package/lib/cli/db/unlock.js +0 -15
- package/lib/cli/index.js +0 -90
- package/lib/controllers/AdminController.js +0 -258
- package/lib/controllers/CollectionController.js +0 -263
- package/lib/controllers/Controller.js +0 -462
- package/lib/controllers/ControllerAction.js +0 -276
- package/lib/controllers/MemberAction.js +0 -22
- package/lib/controllers/ModelController.js +0 -64
- package/lib/controllers/RelationController.js +0 -82
- package/lib/controllers/UserController.js +0 -98
- package/lib/controllers/index.js +0 -50
- package/lib/decorators/action.js +0 -14
- package/lib/decorators/authorize.js +0 -13
- package/lib/decorators/index.js +0 -58
- package/lib/decorators/parameters.js +0 -35
- package/lib/decorators/returns.js +0 -26
- package/lib/decorators/scope.js +0 -14
- package/lib/decorators/transacted.js +0 -12
- package/lib/errors/AssetError.js +0 -19
- package/lib/errors/AuthenticationError.js +0 -19
- package/lib/errors/AuthorizationError.js +0 -19
- package/lib/errors/ControllerError.js +0 -24
- package/lib/errors/DatabaseError.js +0 -27
- package/lib/errors/GraphError.js +0 -19
- package/lib/errors/ModelError.js +0 -24
- package/lib/errors/NotFoundError.js +0 -19
- package/lib/errors/NotImplementedError.js +0 -19
- package/lib/errors/QueryBuilderError.js +0 -19
- package/lib/errors/RelationError.js +0 -36
- package/lib/errors/ResponseError.js +0 -46
- package/lib/errors/ValidationError.js +0 -19
- package/lib/errors/WrappedError.js +0 -24
- package/lib/errors/index.js +0 -122
- package/lib/graph/DitoGraphProcessor.js +0 -185
- package/lib/graph/expression.js +0 -76
- package/lib/graph/graph.js +0 -300
- package/lib/graph/index.js +0 -34
- package/lib/index.js +0 -82
- package/lib/lib/EventEmitter.js +0 -76
- package/lib/lib/KnexHelper.js +0 -40
- package/lib/lib/index.js +0 -26
- package/lib/middleware/attachLogger.js +0 -16
- package/lib/middleware/createTransaction.js +0 -36
- package/lib/middleware/findRoute.js +0 -35
- package/lib/middleware/handleError.js +0 -26
- package/lib/middleware/handleRoute.js +0 -29
- package/lib/middleware/handleUser.js +0 -36
- package/lib/middleware/index.js +0 -66
- package/lib/middleware/logRequests.js +0 -122
- package/lib/mixins/AssetMixin.js +0 -81
- package/lib/mixins/SessionMixin.js +0 -22
- package/lib/mixins/TimeStampedMixin.js +0 -47
- package/lib/mixins/UserMixin.js +0 -151
- package/lib/mixins/index.js +0 -42
- package/lib/models/AssetModel.js +0 -12
- package/lib/models/Model.js +0 -953
- package/lib/models/RelationAccessor.js +0 -41
- package/lib/models/SessionModel.js +0 -12
- package/lib/models/TimeStampedModel.js +0 -12
- package/lib/models/UserModel.js +0 -12
- package/lib/models/definitions/assets.js +0 -11
- package/lib/models/definitions/filters.js +0 -101
- package/lib/models/definitions/hooks.js +0 -11
- package/lib/models/definitions/index.js +0 -38
- package/lib/models/definitions/modifiers.js +0 -11
- package/lib/models/definitions/options.js +0 -11
- package/lib/models/definitions/properties.js +0 -87
- package/lib/models/definitions/relations.js +0 -11
- package/lib/models/definitions/schema.js +0 -11
- package/lib/models/definitions/scopes.js +0 -51
- package/lib/models/index.js +0 -50
- package/lib/query/QueryBuilder.js +0 -745
- package/lib/query/QueryFilters.js +0 -82
- package/lib/query/QueryParameters.js +0 -77
- package/lib/query/Registry.js +0 -40
- package/lib/query/index.js +0 -34
- package/lib/schema/formats/_empty.js +0 -10
- package/lib/schema/formats/_required.js +0 -10
- package/lib/schema/formats/index.js +0 -26
- package/lib/schema/index.js +0 -49
- package/lib/schema/keywords/_computed.js +0 -11
- package/lib/schema/keywords/_foreign.js +0 -11
- package/lib/schema/keywords/_hidden.js +0 -11
- package/lib/schema/keywords/_index.js +0 -11
- package/lib/schema/keywords/_instanceof.js +0 -49
- package/lib/schema/keywords/_primary.js +0 -11
- package/lib/schema/keywords/_range.js +0 -26
- package/lib/schema/keywords/_relate.js +0 -19
- package/lib/schema/keywords/_specificType.js +0 -11
- package/lib/schema/keywords/_unique.js +0 -11
- package/lib/schema/keywords/_unsigned.js +0 -11
- package/lib/schema/keywords/_validate.js +0 -80
- package/lib/schema/keywords/index.js +0 -106
- package/lib/schema/properties.js +0 -227
- package/lib/schema/properties.test.js +0 -573
- package/lib/schema/relations.js +0 -274
- package/lib/schema/relations.test.js +0 -155
- package/lib/services/Service.js +0 -34
- package/lib/services/index.js +0 -18
- package/lib/storage/AssetFile.js +0 -97
- package/lib/storage/DiskStorage.js +0 -125
- package/lib/storage/S3Storage.js +0 -171
- package/lib/storage/Storage.js +0 -209
- package/lib/storage/index.js +0 -34
- package/lib/utils/decorator.js +0 -16
- package/lib/utils/deprecate.js +0 -46
- package/lib/utils/emitter.js +0 -13
- package/lib/utils/function.js +0 -14
- package/lib/utils/function.test.js +0 -49
- package/lib/utils/index.js +0 -66
- package/lib/utils/json.js +0 -9
- package/lib/utils/object.js +0 -92
- package/lib/utils/object.test.js +0 -65
- package/lib/utils/scope.js +0 -14
|
@@ -1,462 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
|
|
3
|
-
exports.__esModule = true;
|
|
4
|
-
exports.Controller = void 0;
|
|
5
|
-
|
|
6
|
-
require("core-js/modules/esnext.async-iterator.for-each.js");
|
|
7
|
-
|
|
8
|
-
require("core-js/modules/esnext.iterator.constructor.js");
|
|
9
|
-
|
|
10
|
-
require("core-js/modules/esnext.iterator.for-each.js");
|
|
11
|
-
|
|
12
|
-
require("core-js/modules/esnext.async-iterator.map.js");
|
|
13
|
-
|
|
14
|
-
require("core-js/modules/esnext.iterator.map.js");
|
|
15
|
-
|
|
16
|
-
require("core-js/modules/esnext.async-iterator.filter.js");
|
|
17
|
-
|
|
18
|
-
require("core-js/modules/esnext.iterator.filter.js");
|
|
19
|
-
|
|
20
|
-
require("core-js/modules/esnext.async-iterator.reduce.js");
|
|
21
|
-
|
|
22
|
-
require("core-js/modules/esnext.iterator.reduce.js");
|
|
23
|
-
|
|
24
|
-
require("core-js/modules/esnext.async-iterator.find.js");
|
|
25
|
-
|
|
26
|
-
require("core-js/modules/esnext.iterator.find.js");
|
|
27
|
-
|
|
28
|
-
require("core-js/modules/esnext.weak-map.delete-all.js");
|
|
29
|
-
|
|
30
|
-
require("core-js/modules/esnext.weak-map.emplace.js");
|
|
31
|
-
|
|
32
|
-
var _picocolors = _interopRequireDefault(require("picocolors"));
|
|
33
|
-
|
|
34
|
-
var _utils = require("../utils");
|
|
35
|
-
|
|
36
|
-
var _lib = require("../lib");
|
|
37
|
-
|
|
38
|
-
var _ControllerAction = _interopRequireDefault(require("./ControllerAction"));
|
|
39
|
-
|
|
40
|
-
var _MemberAction = _interopRequireDefault(require("./MemberAction"));
|
|
41
|
-
|
|
42
|
-
var _errors = require("../errors");
|
|
43
|
-
|
|
44
|
-
var _utils2 = require("@ditojs/utils");
|
|
45
|
-
|
|
46
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
47
|
-
|
|
48
|
-
class Controller {
|
|
49
|
-
constructor(app, namespace) {
|
|
50
|
-
this.app = app;
|
|
51
|
-
this.namespace = this.namespace || namespace;
|
|
52
|
-
this.logging = this.app.config.log.routes;
|
|
53
|
-
this.level = 0;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
initialize() {}
|
|
57
|
-
|
|
58
|
-
setup(isRoot = true, setupActionsObject = true) {
|
|
59
|
-
this._setupEmitter(this.hooks, {
|
|
60
|
-
wildcard: true
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
this.name = this.name || this.constructor.name.match(/^(.*?)(?:Controller|)$/)[1];
|
|
64
|
-
|
|
65
|
-
if (this.path === undefined) {
|
|
66
|
-
this.path = this.app.normalizePath(this.name);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
this.transacted = !!this.transacted;
|
|
70
|
-
|
|
71
|
-
if (isRoot) {
|
|
72
|
-
const {
|
|
73
|
-
path,
|
|
74
|
-
namespace
|
|
75
|
-
} = this;
|
|
76
|
-
const url = path ? `/${path}` : '';
|
|
77
|
-
this.url = namespace ? `/${namespace}${url}` : url;
|
|
78
|
-
this.log(`${namespace ? _picocolors.default.green(`/${namespace}/`) : ''}${_picocolors.default.cyan(path)}${_picocolors.default.white(':')}`, this.level);
|
|
79
|
-
|
|
80
|
-
if (setupActionsObject) {
|
|
81
|
-
this.actions = this.actions || this.reflectActionsObject();
|
|
82
|
-
this.actions = this.setupActions('actions');
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
this.assets = this.setupAssets();
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
reflectActionsObject() {
|
|
90
|
-
const {
|
|
91
|
-
allow
|
|
92
|
-
} = this;
|
|
93
|
-
const controller = allow ? {
|
|
94
|
-
allow
|
|
95
|
-
} : {};
|
|
96
|
-
|
|
97
|
-
const addAction = key => {
|
|
98
|
-
const value = this[key];
|
|
99
|
-
|
|
100
|
-
if (value != null && value.verb) {
|
|
101
|
-
controller[key] = value;
|
|
102
|
-
}
|
|
103
|
-
};
|
|
104
|
-
|
|
105
|
-
const proto = Object.getPrototypeOf(this);
|
|
106
|
-
Object.getOwnPropertyNames(proto).forEach(addAction);
|
|
107
|
-
Object.getOwnPropertyNames(this).forEach(addAction);
|
|
108
|
-
return controller;
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
setupRoute(verb, url, transacted, authorize, action, handlers) {
|
|
112
|
-
this.log(`${_picocolors.default.magenta(verb.toUpperCase())} ${_picocolors.default.green(this.url)}${_picocolors.default.cyan(url.slice(this.url.length))} ${_picocolors.default.white(this.describeAuthorize(authorize))}`, this.level + 1);
|
|
113
|
-
this.app.addRoute(verb, url, transacted, handlers, this, action);
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
setupActions(type) {
|
|
117
|
-
const {
|
|
118
|
-
values: actions,
|
|
119
|
-
authorize
|
|
120
|
-
} = this.processValues(this.inheritValues(type));
|
|
121
|
-
|
|
122
|
-
for (const [name, handler] of Object.entries(actions)) {
|
|
123
|
-
this.setupAction(type, actions, name, handler, authorize[name]);
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
return actions;
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
setupAction(type, actions, name, handler, authorize, verb = 'get', path = this.app.normalizePath(name)) {
|
|
130
|
-
if (!(0, _utils2.isFunction)(handler)) {
|
|
131
|
-
handler = setupHandlerFromObject(handler, actions);
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
const actionClass = type === 'member' ? _MemberAction.default : _ControllerAction.default;
|
|
135
|
-
this.setupActionRoute(type, new actionClass(this, handler, type, name, verb, path, authorize));
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
setupActionRoute(type, action) {
|
|
139
|
-
const url = this.getUrl(type, action.path);
|
|
140
|
-
const {
|
|
141
|
-
verb,
|
|
142
|
-
transacted,
|
|
143
|
-
authorize
|
|
144
|
-
} = action;
|
|
145
|
-
this.setupRoute(verb, url, transacted, authorize, action, [async ctx => {
|
|
146
|
-
try {
|
|
147
|
-
const res = await action.callAction(ctx);
|
|
148
|
-
|
|
149
|
-
if (res !== undefined) {
|
|
150
|
-
ctx.body = res;
|
|
151
|
-
}
|
|
152
|
-
} catch (err) {
|
|
153
|
-
throw err instanceof _errors.ResponseError ? err : new _errors.WrappedError(err);
|
|
154
|
-
}
|
|
155
|
-
}]);
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
setupAssets() {
|
|
159
|
-
const {
|
|
160
|
-
values: assets,
|
|
161
|
-
authorize
|
|
162
|
-
} = this.processValues(this.inheritValues('assets'));
|
|
163
|
-
|
|
164
|
-
for (const [dataPath, config] of Object.entries(assets || {})) {
|
|
165
|
-
this.setupAssetRoute(dataPath, config, authorize[dataPath]);
|
|
166
|
-
}
|
|
167
|
-
|
|
168
|
-
return assets;
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
setupAssetRoute(dataPath, config, authorize) {
|
|
172
|
-
const {
|
|
173
|
-
storage: storageName,
|
|
174
|
-
transacted,
|
|
175
|
-
...settings
|
|
176
|
-
} = config;
|
|
177
|
-
const storage = this.app.getStorage(storageName);
|
|
178
|
-
|
|
179
|
-
if (!storage) {
|
|
180
|
-
throw new _errors.ControllerError(this, `Unknown storage configuration: '${storageName}'`);
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
const tokens = (0, _utils2.parseDataPath)(dataPath);
|
|
184
|
-
|
|
185
|
-
const getDataPath = callback => (0, _utils2.normalizeDataPath)(tokens.map(callback));
|
|
186
|
-
|
|
187
|
-
let index = 0;
|
|
188
|
-
const multipleWildcards = tokens.filter(token => token === '*').length > 1;
|
|
189
|
-
const normalizedPath = getDataPath(token => token === '*' ? multipleWildcards ? `:index${++index}` : ':index' : this.app.normalizePath(token));
|
|
190
|
-
const matchDataPath = new RegExp(`^${getDataPath(token => token === '*' ? '\\w+' : token)}$`);
|
|
191
|
-
const url = this.getUrl('controller', `upload/${normalizedPath}`);
|
|
192
|
-
const upload = storage.getUploadHandler({ ...settings,
|
|
193
|
-
fileFilter: (req, file, cb) => {
|
|
194
|
-
cb(null, matchDataPath.test(file.fieldname));
|
|
195
|
-
}
|
|
196
|
-
});
|
|
197
|
-
const authorization = this.processAuthorize(authorize);
|
|
198
|
-
this.setupRoute('post', url, transacted, authorize, null, [async (ctx, next) => {
|
|
199
|
-
await this.handleAuthorization(authorization, ctx);
|
|
200
|
-
return next();
|
|
201
|
-
}, upload, async (ctx, next) => {
|
|
202
|
-
const files = storage.convertStorageFiles(ctx.request.files);
|
|
203
|
-
await this.app.createAssets(storage, files, 0, ctx.transaction);
|
|
204
|
-
ctx.body = files;
|
|
205
|
-
return next();
|
|
206
|
-
}]);
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
compose() {}
|
|
210
|
-
|
|
211
|
-
getPath(type, path) {
|
|
212
|
-
return path;
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
getUrl(type, path) {
|
|
216
|
-
path = this.getPath(type, path);
|
|
217
|
-
return path && path !== '.' ? `${this.url}/${path}` : this.url;
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
inheritValues(type) {
|
|
221
|
-
const parentClass = Object.getPrototypeOf(this.constructor);
|
|
222
|
-
|
|
223
|
-
if (!inheritanceMap.has(parentClass)) {
|
|
224
|
-
inheritanceMap.set(parentClass, {
|
|
225
|
-
instance: new parentClass(this.app, this.namespace)
|
|
226
|
-
});
|
|
227
|
-
}
|
|
228
|
-
|
|
229
|
-
const entry = inheritanceMap.get(parentClass);
|
|
230
|
-
|
|
231
|
-
if (!entry[type]) {
|
|
232
|
-
const parent = entry.instance;
|
|
233
|
-
let values = parent[type];
|
|
234
|
-
|
|
235
|
-
if (parentClass !== Controller) {
|
|
236
|
-
values = parent.inheritValues(type);
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
entry[type] = values;
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
const parentValues = entry[type];
|
|
243
|
-
let currentValues = this[type];
|
|
244
|
-
|
|
245
|
-
if (currentValues && currentValues === parentValues) {
|
|
246
|
-
currentValues = this[type] = {};
|
|
247
|
-
}
|
|
248
|
-
|
|
249
|
-
return (0, _utils2.isObject)(parentValues) && (0, _utils2.isObject)(currentValues) ? Object.setPrototypeOf(currentValues, parentValues) : currentValues;
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
processValues(values) {
|
|
253
|
-
if (!values) return {};
|
|
254
|
-
const mergedAllow = {};
|
|
255
|
-
const mergedAuthorize = {};
|
|
256
|
-
let hasOwnAllow = false;
|
|
257
|
-
|
|
258
|
-
const excludeKey = key => ['allow', 'authorize'].includes(key);
|
|
259
|
-
|
|
260
|
-
const handleAllow = (allow, current) => {
|
|
261
|
-
if (allow) {
|
|
262
|
-
allow = (0, _utils2.asArray)(allow);
|
|
263
|
-
hasOwnAllow = true;
|
|
264
|
-
} else if (!hasOwnAllow) {
|
|
265
|
-
allow = Object.keys(current);
|
|
266
|
-
}
|
|
267
|
-
|
|
268
|
-
if (allow) {
|
|
269
|
-
if (allow.includes('*')) {
|
|
270
|
-
allow = (0, _utils.getAllKeys)(current);
|
|
271
|
-
}
|
|
272
|
-
|
|
273
|
-
for (const key of allow) {
|
|
274
|
-
if (!excludeKey(key)) {
|
|
275
|
-
mergedAllow[key] = true;
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
}
|
|
279
|
-
};
|
|
280
|
-
|
|
281
|
-
const handleAuthorize = authorize => {
|
|
282
|
-
const add = (key, value) => {
|
|
283
|
-
if (key in values && !(key in mergedAuthorize) && !excludeKey(key)) {
|
|
284
|
-
mergedAuthorize[key] = value;
|
|
285
|
-
}
|
|
286
|
-
};
|
|
287
|
-
|
|
288
|
-
if ((0, _utils2.isObject)(authorize)) {
|
|
289
|
-
for (const key in authorize) {
|
|
290
|
-
add(key, authorize[key]);
|
|
291
|
-
}
|
|
292
|
-
} else if (authorize != null) {
|
|
293
|
-
for (const key in values) {
|
|
294
|
-
add(key, authorize);
|
|
295
|
-
}
|
|
296
|
-
}
|
|
297
|
-
};
|
|
298
|
-
|
|
299
|
-
let current = values;
|
|
300
|
-
|
|
301
|
-
while (current !== Object.prototype && !current.hasOwnProperty('$core')) {
|
|
302
|
-
handleAllow((0, _utils.getOwnProperty)(current, 'allow'), current);
|
|
303
|
-
handleAuthorize((0, _utils.getOwnProperty)(current, 'authorize'));
|
|
304
|
-
current = Object.getPrototypeOf(current);
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
if (this.allow) {
|
|
308
|
-
handleAllow(this.allow, values);
|
|
309
|
-
}
|
|
310
|
-
|
|
311
|
-
if (this.authorize) {
|
|
312
|
-
handleAuthorize(this.authorize);
|
|
313
|
-
}
|
|
314
|
-
|
|
315
|
-
return {
|
|
316
|
-
values: (0, _utils.getAllKeys)(values).reduce((result, key) => {
|
|
317
|
-
if (mergedAllow[key]) {
|
|
318
|
-
result[key] = values[key];
|
|
319
|
-
}
|
|
320
|
-
|
|
321
|
-
return result;
|
|
322
|
-
}, Object.create(Object.getPrototypeOf(values))),
|
|
323
|
-
allow: Object.keys(mergedAllow),
|
|
324
|
-
authorize: mergedAuthorize
|
|
325
|
-
};
|
|
326
|
-
}
|
|
327
|
-
|
|
328
|
-
async emitHook(type, handleResult, ctx, ...args) {
|
|
329
|
-
let result = handleResult ? args.shift() : undefined;
|
|
330
|
-
|
|
331
|
-
for (const handler of this.listeners(type)) {
|
|
332
|
-
if (handleResult) {
|
|
333
|
-
const res = await handler.call(this, ctx, result, ...args);
|
|
334
|
-
|
|
335
|
-
if (res !== undefined) {
|
|
336
|
-
result = res;
|
|
337
|
-
}
|
|
338
|
-
} else {
|
|
339
|
-
await handler.call(this, ctx, ...args);
|
|
340
|
-
}
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
return result;
|
|
344
|
-
}
|
|
345
|
-
|
|
346
|
-
async getMember() {
|
|
347
|
-
return null;
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
processAuthorize(authorize) {
|
|
351
|
-
if (authorize == null) {
|
|
352
|
-
return () => true;
|
|
353
|
-
} else if ((0, _utils2.isBoolean)(authorize)) {
|
|
354
|
-
return () => authorize;
|
|
355
|
-
} else if ((0, _utils2.isFunction)(authorize)) {
|
|
356
|
-
return async (ctx, member) => {
|
|
357
|
-
const res = await authorize(ctx, member);
|
|
358
|
-
return this.processAuthorize(res)(ctx, member);
|
|
359
|
-
};
|
|
360
|
-
} else if ((0, _utils2.isString)(authorize) || (0, _utils2.isArray)(authorize)) {
|
|
361
|
-
return async (ctx, member) => {
|
|
362
|
-
const {
|
|
363
|
-
user
|
|
364
|
-
} = ctx.state;
|
|
365
|
-
|
|
366
|
-
if (!user) {
|
|
367
|
-
return false;
|
|
368
|
-
}
|
|
369
|
-
|
|
370
|
-
const values = (0, _utils2.asArray)(authorize);
|
|
371
|
-
|
|
372
|
-
if (!member && values.includes('$owner')) {
|
|
373
|
-
member = await this.getMember(ctx);
|
|
374
|
-
}
|
|
375
|
-
|
|
376
|
-
return !!values.find(value => {
|
|
377
|
-
var _member;
|
|
378
|
-
|
|
379
|
-
return value === '$self' ? user.constructor === this.modelClass && (0, _utils2.equals)(user.$id(), ctx.memberId) : value === '$owner' ? (_member = member) == null ? void 0 : _member.$hasOwner == null ? void 0 : _member.$hasOwner(user) : user.$hasRole(value);
|
|
380
|
-
});
|
|
381
|
-
};
|
|
382
|
-
} else {
|
|
383
|
-
throw new _errors.ControllerError(this, `Unsupported authorize setting: '${authorize}'`);
|
|
384
|
-
}
|
|
385
|
-
}
|
|
386
|
-
|
|
387
|
-
describeAuthorize(authorize) {
|
|
388
|
-
return (0, _utils2.isFunction)(authorize) ? (0, _utils.describeFunction)(authorize) : (0, _utils2.isString)(authorize) ? `'${authorize}'` : (0, _utils2.isArray)(authorize) ? `[${authorize.map(value => `'${value}'`).join(', ')}]` : '';
|
|
389
|
-
}
|
|
390
|
-
|
|
391
|
-
async handleAuthorization(authorization, ctx, member) {
|
|
392
|
-
const ok = await authorization(ctx, member);
|
|
393
|
-
|
|
394
|
-
if (ok !== true) {
|
|
395
|
-
throw new _errors.AuthorizationError();
|
|
396
|
-
}
|
|
397
|
-
}
|
|
398
|
-
|
|
399
|
-
log(str, indent = 0) {
|
|
400
|
-
if (this.logging) {
|
|
401
|
-
console.info(`${' '.repeat(indent)}${str}`);
|
|
402
|
-
}
|
|
403
|
-
}
|
|
404
|
-
|
|
405
|
-
}
|
|
406
|
-
|
|
407
|
-
exports.Controller = Controller;
|
|
408
|
-
|
|
409
|
-
_lib.EventEmitter.mixin(Controller.prototype);
|
|
410
|
-
|
|
411
|
-
const inheritanceMap = new WeakMap();
|
|
412
|
-
|
|
413
|
-
function setupHandlerFromObject(object, actions) {
|
|
414
|
-
const {
|
|
415
|
-
handler,
|
|
416
|
-
action,
|
|
417
|
-
authorize,
|
|
418
|
-
parameters,
|
|
419
|
-
returns,
|
|
420
|
-
scope,
|
|
421
|
-
transacted
|
|
422
|
-
} = object;
|
|
423
|
-
Object.setPrototypeOf(object, Object.getPrototypeOf(actions));
|
|
424
|
-
handler.authorize = authorize;
|
|
425
|
-
handler.transacted = transacted;
|
|
426
|
-
|
|
427
|
-
if (action) {
|
|
428
|
-
const [verb, path] = (0, _utils2.asArray)(action);
|
|
429
|
-
handler.verb = verb;
|
|
430
|
-
handler.path = path;
|
|
431
|
-
}
|
|
432
|
-
|
|
433
|
-
if (parameters) {
|
|
434
|
-
const [_parameters, options] = parameters;
|
|
435
|
-
const hasOptions = (0, _utils2.isArray)(_parameters);
|
|
436
|
-
handler.parameters = hasOptions ? _parameters : parameters;
|
|
437
|
-
|
|
438
|
-
if (hasOptions) {
|
|
439
|
-
handler.options = { ...handler.options,
|
|
440
|
-
parameters: options
|
|
441
|
-
};
|
|
442
|
-
}
|
|
443
|
-
}
|
|
444
|
-
|
|
445
|
-
if (returns) {
|
|
446
|
-
const [_returns, options] = (0, _utils2.asArray)(returns);
|
|
447
|
-
handler.returns = _returns;
|
|
448
|
-
|
|
449
|
-
if (options) {
|
|
450
|
-
handler.options = { ...handler.options,
|
|
451
|
-
parameters: options
|
|
452
|
-
};
|
|
453
|
-
}
|
|
454
|
-
}
|
|
455
|
-
|
|
456
|
-
if (scope) {
|
|
457
|
-
handler.scope = (0, _utils2.asArray)(scope);
|
|
458
|
-
}
|
|
459
|
-
|
|
460
|
-
return handler;
|
|
461
|
-
}
|
|
462
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/controllers/Controller.js"],"names":["Controller","constructor","app","namespace","logging","config","log","routes","level","initialize","setup","isRoot","setupActionsObject","_setupEmitter","hooks","wildcard","name","match","path","undefined","normalizePath","transacted","url","pico","green","cyan","white","actions","reflectActionsObject","setupActions","assets","setupAssets","allow","controller","addAction","key","value","verb","proto","Object","getPrototypeOf","getOwnPropertyNames","forEach","setupRoute","authorize","action","handlers","magenta","toUpperCase","slice","length","describeAuthorize","addRoute","type","values","processValues","inheritValues","handler","entries","setupAction","setupHandlerFromObject","actionClass","MemberAction","ControllerAction","setupActionRoute","getUrl","ctx","res","callAction","body","err","ResponseError","WrappedError","dataPath","setupAssetRoute","storage","storageName","settings","getStorage","ControllerError","tokens","getDataPath","callback","map","index","multipleWildcards","filter","token","normalizedPath","matchDataPath","RegExp","upload","getUploadHandler","fileFilter","req","file","cb","test","fieldname","authorization","processAuthorize","next","handleAuthorization","files","convertStorageFiles","request","createAssets","transaction","compose","getPath","parentClass","inheritanceMap","has","set","instance","entry","get","parent","parentValues","currentValues","setPrototypeOf","mergedAllow","mergedAuthorize","hasOwnAllow","excludeKey","includes","handleAllow","current","keys","handleAuthorize","add","prototype","hasOwnProperty","reduce","result","create","emitHook","handleResult","args","shift","listeners","call","getMember","member","user","state","find","modelClass","$id","memberId","$hasOwner","$hasRole","join","ok","AuthorizationError","str","indent","console","info","repeat","EventEmitter","mixin","WeakMap","object","parameters","returns","scope","_parameters","options","hasOptions","_returns"],"mappings":"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","sourcesContent":["import pico from 'picocolors'\nimport { getOwnProperty, getAllKeys, describeFunction } from '@/utils'\nimport { EventEmitter } from '@/lib'\nimport ControllerAction from './ControllerAction'\nimport MemberAction from './MemberAction'\nimport {\n  ResponseError, WrappedError, ControllerError, AuthorizationError\n} from '@/errors'\nimport {\n  isObject, isString, isArray, isBoolean, isFunction, asArray, equals,\n  parseDataPath, normalizeDataPath\n} from '@ditojs/utils'\n\nexport class Controller {\n  constructor(app, namespace) {\n    this.app = app\n    this.namespace = this.namespace || namespace\n    this.logging = this.app.config.log.routes\n    this.level = 0\n  }\n\n  // @overridable\n  initialize() {\n  }\n\n  setup(isRoot = true, setupActionsObject = true) {\n    this._setupEmitter(this.hooks, {\n      // Support wildcard hooks only on controllers:\n      wildcard: true\n    })\n    // If the class name ends in 'Controller', remove it from controller name.\n    this.name = this.name ||\n      this.constructor.name.match(/^(.*?)(?:Controller|)$/)[1]\n    if (this.path === undefined) {\n      this.path = this.app.normalizePath(this.name)\n    }\n    this.transacted = !!this.transacted\n    if (isRoot) {\n      const { path, namespace } = this\n      // TODO: The distinction between  `url` and `path` is a bit tricky, since\n      // what we call `url` here is called `path` in Router, and may contain\n      // mapped parameters or wildcards. Consider `path` / `route` instead?\n      const url = path ? `/${path}` : ''\n      this.url = namespace ? `/${namespace}${url}` : url\n      this.log(\n        `${\n          namespace ? pico.green(`/${namespace}/`) : ''\n        }${\n          pico.cyan(path)\n        }${\n          pico.white(':')\n        }`,\n        this.level\n      )\n      if (setupActionsObject) {\n        this.actions = this.actions || this.reflectActionsObject()\n        // Now that the instance fields are reflected in the `controller` object\n        // we can use the normal inheritance mechanism through `setupActions()`:\n        this.actions = this.setupActions('actions')\n      }\n      this.assets = this.setupAssets()\n    }\n  }\n\n  reflectActionsObject() {\n    // On base controllers, the actions can be defined directly in the class\n    // instead of inside an actions object, as is done with model and relation\n    // controllers. But in order to use the same structure for inheritance as\n    // these other controllers, we reflect these instance fields in a separate\n    // `actions` object.\n    const { allow } = this\n    const controller = allow ? { allow } : {}\n\n    const addAction = key => {\n      const value = this[key]\n      // NOTE: Only add instance methods that have a @action() decorator,\n      // which in turn sets the `verb` property on the method, or action objects\n      // which have the `verb` property:\n      if (value?.verb) {\n        controller[key] = value\n      }\n    }\n    // Use `Object.getOwnPropertyNames()` to get the fields, in order to\n    // not also receive values from parents (those are fetched later in\n    // `inheritValues()`, see `getParentValues()`).\n    const proto = Object.getPrototypeOf(this)\n    Object.getOwnPropertyNames(proto).forEach(addAction)\n    Object.getOwnPropertyNames(this).forEach(addAction)\n    return controller\n  }\n\n  setupRoute(verb, url, transacted, authorize, action, handlers) {\n    this.log(\n      `${\n        pico.magenta(verb.toUpperCase())\n      } ${\n        pico.green(this.url)\n      }${\n        pico.cyan(url.slice(this.url.length))\n      } ${\n        pico.white(this.describeAuthorize(authorize))\n      }`,\n      this.level + 1\n    )\n    this.app.addRoute(verb, url, transacted, handlers, this, action)\n  }\n\n  setupActions(type) {\n    const {\n      values: actions,\n      authorize\n    } = this.processValues(this.inheritValues(type))\n    for (const [name, handler] of Object.entries(actions)) {\n      this.setupAction(type, actions, name, handler, authorize[name])\n    }\n    return actions\n  }\n\n  setupAction(\n    type,\n    actions,\n    name,\n    handler,\n    authorize,\n    // These values are only changed when called from\n    // `CollectionController.setupAction()`:\n    verb = 'get',\n    // The default path for actions is the normalized name.\n    path = this.app.normalizePath(name)\n  ) {\n    if (!isFunction(handler)) {\n      handler = setupHandlerFromObject(handler, actions)\n    }\n    // Custom member actions have their own class so they can fetch the members\n    // ahead of their call.\n    const actionClass = type === 'member' ? MemberAction : ControllerAction\n    this.setupActionRoute(\n      type,\n      // eslint-disable-next-line new-cap\n      new actionClass(this, handler, type, name, verb, path, authorize)\n    )\n  }\n\n  setupActionRoute(type, action) {\n    const url = this.getUrl(type, action.path)\n    const { verb, transacted, authorize } = action\n    this.setupRoute(verb, url, transacted, authorize, action, [\n      async ctx => {\n        try {\n          const res = await action.callAction(ctx)\n          if (res !== undefined) {\n            ctx.body = res\n          }\n        } catch (err) {\n          throw err instanceof ResponseError ? err : new WrappedError(err)\n        }\n      }\n    ])\n  }\n\n  setupAssets() {\n    const {\n      values: assets,\n      authorize\n    } = this.processValues(this.inheritValues('assets'))\n    for (const [dataPath, config] of Object.entries(assets || {})) {\n      this.setupAssetRoute(dataPath, config, authorize[dataPath])\n    }\n    return assets\n  }\n\n  setupAssetRoute(dataPath, config, authorize) {\n    const {\n      storage: storageName,\n      // TODO: What exactly should control the use of `transacted`?\n      transacted,\n      ...settings\n    } = config\n    const storage = this.app.getStorage(storageName)\n    if (!storage) {\n      throw new ControllerError(this,\n        `Unknown storage configuration: '${storageName}'`\n      )\n    }\n    const tokens = parseDataPath(dataPath)\n    const getDataPath = callback => normalizeDataPath(tokens.map(callback))\n\n    // Replace wildcards with numbered indices and convert to '/'-notation:\n    let index = 0\n    const multipleWildcards = tokens.filter(token => token === '*').length > 1\n    const normalizedPath = getDataPath(\n      token => token === '*'\n        ? multipleWildcards\n          ? `:index${++index}`\n          : ':index'\n        : this.app.normalizePath(token)\n    )\n\n    // Convert `dataPath` to a regular expression to match field names\n    // against, but convert wildcards (*) to match both numeric ids and words,\n    // e.g. 'create':\n    const matchDataPath = new RegExp(\n      `^${getDataPath(token => token === '*' ? '\\\\w+' : token)}$`\n    )\n\n    const url = this.getUrl('controller', `upload/${normalizedPath}`)\n    const upload = storage.getUploadHandler({\n      ...settings,\n      // Only let uploads pass that match the normalizePath + wildcards:\n      fileFilter: (req, file, cb) => {\n        cb(null, matchDataPath.test(file.fieldname))\n      }\n    })\n\n    const authorization = this.processAuthorize(authorize)\n    this.setupRoute('post', url, transacted, authorize, null, [\n      async (ctx, next) => {\n        await this.handleAuthorization(authorization, ctx)\n        return next()\n      },\n\n      upload,\n\n      async (ctx, next) => {\n        const files = storage.convertStorageFiles(ctx.request.files)\n        await this.app.createAssets(storage, files, 0, ctx.transaction)\n        // Send the file objects back for the upload component to store in the\n        // data.\n        ctx.body = files\n        return next()\n      }\n    ])\n  }\n\n  compose() {\n    // To be overridden in sub-classes, if the controller needs to install\n    // middleware. For normal routes, use `this.app.addRoute()` instead.\n  }\n\n  getPath(type, path) {\n    // To be overridden by sub-classes.\n    return path\n  }\n\n  getUrl(type, path) {\n    path = this.getPath(type, path)\n    // Use '.' as the path for the controller's \"index\" action.\n    return path && path !== '.' ? `${this.url}/${path}` : this.url\n  }\n\n  inheritValues(type) {\n    // Gets the controller class's instance field for a given action type, e.g.\n    // `controller` (Controller), `collection`, `member` (ModelController,\n    // RelationController), `relation` (RelationController), and sets up an\n    // inheritance chain for it that goes all the way up to it base class (e.g.\n    // CollectionController), so that the default definitions for all http verbs\n    // can be correctly inherited and overridden while using `super.<action>()`.\n    const parentClass = Object.getPrototypeOf(this.constructor)\n    // Create one instance of each controller class up the chain in order to\n    // get to their definitions of the inheritable values. Cache both instance\n    // and resolved values per parentClass in an inheritanceMap.\n    if (!inheritanceMap.has(parentClass)) {\n      inheritanceMap.set(parentClass, {\n        // eslint-disable-next-line new-cap\n        instance: new parentClass(this.app, this.namespace)\n      })\n    }\n    const entry = inheritanceMap.get(parentClass)\n    if (!entry[type]) {\n      const parent = entry.instance\n      let values = parent[type]\n      if (parentClass !== Controller) {\n        // Recursively set up inheritance chains.\n        values = parent.inheritValues(type)\n      }\n      entry[type] = values\n    }\n    // If there are no values defined on `this` that differ from the parent,\n    // set to an empty object so inheritance can be set up and `processValues()`\n    // can still be called.\n    // NOTE: We can't check with `this.hasOwnProperty(type)` because the\n    // field can be on the class prototype as well, in case of accessors.\n    const parentValues = entry[type]\n    let currentValues = this[type]\n    if (currentValues && currentValues === parentValues) {\n      currentValues = this[type] = {}\n    }\n    // Combine parentValues and currentValues with correct inheritance.\n    return isObject(parentValues) && isObject(currentValues)\n      ? Object.setPrototypeOf(currentValues, parentValues)\n      : currentValues\n  }\n\n  processValues(values) {\n    if (!values) return {}\n    // Respect `allow` settings and clear entries that aren't allowed.\n    // Also collect and expand `authorize` settings so that in the end, an\n    // `authorize` object can be returned with valid settings for all values.\n    //\n    // Rules:\n    // - Own values on objects that don't define an `allow` array are\n    //   automatically allowed. If an `allow` array is defined as well, then\n    //   these own values need to be explicitly listed.\n    // - If no `allow` arrays are defined in the prototypal hierarchy, each\n    //   level allows its own values, and these are merged, except for those\n    //   marked as `$core`, which need to be explicitly listed in `allow`.\n\n    // NOTE: `handleAllow()` and `handleAuthorize()` are applied in sequence of\n    // the `values` inheritance, from sub-class to base-class.\n\n    const mergedAllow = {}\n    const mergedAuthorize = {}\n    let hasOwnAllow = false\n\n    const excludeKey = key => ['allow', 'authorize'].includes(key)\n\n    const handleAllow = (allow, current) => {\n      if (allow) {\n        allow = asArray(allow)\n        hasOwnAllow = true\n      } else if (!hasOwnAllow) {\n        // Only keep adding to the merged `allow` if we didn't already encounter\n        // an own `allow` object further up the chain.\n        allow = Object.keys(current)\n      }\n      if (allow) {\n        if (allow.includes('*')) {\n          allow = getAllKeys(current)\n        }\n        for (const key of allow) {\n          if (!excludeKey(key)) {\n            mergedAllow[key] = true\n          }\n        }\n      }\n    }\n\n    const handleAuthorize = authorize => {\n      const add = (key, value) => {\n        // Since we're walking up in the inheritance chain, only take on an\n        // authorize setting for a given key if it wasn't already defined before\n        if (\n          key in values &&\n          !(key in mergedAuthorize) &&\n          !excludeKey(key)\n        ) {\n          mergedAuthorize[key] = value\n        }\n      }\n\n      if (isObject(authorize)) {\n        for (const key in authorize) {\n          add(key, authorize[key])\n        }\n      } else if (authorize != null) {\n        // This is a values-wide setting. Loop through all values, not just\n        // current ones, and apply to any action that doesn't already have one:\n        for (const key in values) {\n          add(key, authorize)\n        }\n      }\n    }\n\n    // Process the `allow` and `authorize` settings in sequence of the `values`\n    // inheritance, from sub-class to base-class.\n    let current = values\n    while (current !== Object.prototype && !current.hasOwnProperty('$core')) {\n      handleAllow(getOwnProperty(current, 'allow'), current)\n      handleAuthorize(getOwnProperty(current, 'authorize'))\n      current = Object.getPrototypeOf(current)\n    }\n\n    // At the end of the chain, also support both settings on the controller-\n    // level, and thus applied to all action objects in the controller.\n    if (this.allow) {\n      handleAllow(this.allow, values)\n    }\n    if (this.authorize) {\n      handleAuthorize(this.authorize)\n    }\n\n    return {\n      // Create a filtered `values` object that only contains the allowed fields\n      values: getAllKeys(values).reduce(\n        (result, key) => {\n          if (mergedAllow[key]) {\n            result[key] = values[key]\n          }\n          return result\n        },\n        // Create a new object for the filtered `values` that keeps inheritance\n        // intact. This is required by `setupHandlerFromObject()`, to support\n        // `super` in handler functions.\n        Object.create(Object.getPrototypeOf(values))\n      ),\n      allow: Object.keys(mergedAllow),\n      authorize: mergedAuthorize\n    }\n  }\n\n  async emitHook(type, handleResult, ctx, ...args) {\n    let result = handleResult ? args.shift() : undefined\n    for (const handler of this.listeners(type)) {\n      if (handleResult) {\n        const res = await handler.call(this, ctx, result, ...args)\n        if (res !== undefined) {\n          result = res\n        }\n      } else {\n        await handler.call(this, ctx, ...args)\n      }\n    }\n    return result\n  }\n\n  async getMember(/* ctx, base = this, param = { ... } */) {\n    // This is only defined in `CollectionController`, where it resolves to the\n    // member represented by the given route.\n    return null\n  }\n\n  /**\n   * Converts the authorize config settings into an authorization function that\n   * can be passed to `handleAuthorization()`.\n   *\n   * @param {boolean|function|string|string[]} authorize the authorize config\n   * settings\n   * @return {function} the authorization function\n   */\n  processAuthorize(authorize) {\n    if (authorize == null) {\n      return () => true\n    } else if (isBoolean(authorize)) {\n      return () => authorize\n    } else if (isFunction(authorize)) {\n      return async (ctx, member) => {\n        const res = await authorize(ctx, member)\n        // Pass res through `processAuthorize()` to support strings & arrays.\n        return this.processAuthorize(res)(ctx, member)\n      }\n    } else if (isString(authorize) || isArray(authorize)) {\n      return async (ctx, member) => {\n        const { user } = ctx.state\n        if (!user) {\n          return false\n        }\n        const values = asArray(authorize)\n        // For '$owner', fetch `member` now in case the action parameters\n        // didn't already request it:\n        if (!member && values.includes('$owner')) {\n          member = await this.getMember(ctx)\n        }\n        return !!values.find(\n        // Support 3 scenarios:\n        // - '$self': The requested member is checked against `ctx.state.user`\n        //   and the action is only authorized if it matches the member.\n        // - '$owner': The member is asked if it is owned by `ctx.state.user`\n        //   through the optional `Model.$hasOwner()` method.\n        // - any string:  `ctx.state.user` is checked for this role through\n        //   the overridable `UserModel.hasRole()` method.\n          value => {\n            return value === '$self'\n              ? user.constructor === this.modelClass &&\n              equals(user.$id(), ctx.memberId)\n              : value === '$owner'\n                ? member?.$hasOwner?.(user)\n                : user.$hasRole(value)\n          }\n        )\n      }\n    } else {\n      throw new ControllerError(this,\n        `Unsupported authorize setting: '${authorize}'`\n      )\n    }\n  }\n\n  describeAuthorize(authorize) {\n    return isFunction(authorize)\n      ? describeFunction(authorize)\n      : isString(authorize)\n        ? `'${authorize}'`\n        : isArray(authorize)\n          ? `[${authorize.map(value => `'${value}'`).join(', ')}]`\n          : ''\n  }\n\n  async handleAuthorization(authorization, ctx, member) {\n    const ok = await authorization(ctx, member)\n    if (ok !== true) {\n      throw new AuthorizationError()\n    }\n  }\n\n  log(str, indent = 0) {\n    if (this.logging) {\n      console.info(`${'  '.repeat(indent)}${str}`)\n    }\n  }\n}\n\nEventEmitter.mixin(Controller.prototype)\n\nconst inheritanceMap = new WeakMap()\n\nfunction setupHandlerFromObject(object, actions) {\n  const {\n    handler,\n    action,\n    authorize,\n    parameters,\n    returns,\n    scope,\n    transacted\n  } = object\n\n  // In order to suport `super` calls in the `handler` function in object\n  // notation, deploy this crazy JS sorcery:\n  Object.setPrototypeOf(object, Object.getPrototypeOf(actions))\n\n  handler.authorize = authorize\n  handler.transacted = transacted\n\n  if (action) {\n    const [verb, path] = asArray(action)\n    handler.verb = verb\n    handler.path = path\n  }\n\n  if (parameters) {\n    const [_parameters, options] = parameters\n    const hasOptions = isArray(_parameters)\n    handler.parameters = hasOptions ? _parameters : parameters\n\n    // If validation options are provided, expose them through\n    // `handler.options.parameters`, see ControllerAction\n    if (hasOptions) {\n      handler.options = {\n        ...handler.options,\n        parameters: options\n      }\n    }\n  }\n\n  if (returns) {\n    const [_returns, options] = asArray(returns)\n    handler.returns = _returns\n\n    // If validation options are provided, expose them through\n    // `handler.options.returns`, see ControllerAction\n    if (options) {\n      handler.options = {\n        ...handler.options,\n        parameters: options\n      }\n    }\n  }\n\n  if (scope) {\n    handler.scope = asArray(scope)\n  }\n\n  return handler\n}\n"]}
|