@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.
Files changed (216) hide show
  1. package/package.json +25 -43
  2. package/src/app/Application.js +105 -111
  3. package/src/app/Validator.js +6 -3
  4. package/src/app/index.js +2 -2
  5. package/src/cli/db/createMigration.js +1 -1
  6. package/src/cli/db/index.js +7 -7
  7. package/src/cli/db/listAssetConfig.js +1 -1
  8. package/src/cli/db/reset.js +1 -1
  9. package/src/cli/index.js +14 -5
  10. package/src/controllers/AdminController.js +181 -158
  11. package/src/controllers/CollectionController.js +8 -29
  12. package/src/controllers/Controller.js +71 -76
  13. package/src/controllers/ControllerAction.js +39 -17
  14. package/src/controllers/MemberAction.js +2 -2
  15. package/src/controllers/ModelController.js +4 -4
  16. package/src/controllers/RelationController.js +3 -3
  17. package/src/controllers/{UserController.js → UsersController.js} +8 -13
  18. package/src/controllers/index.js +5 -5
  19. package/src/decorators/action.js +3 -3
  20. package/src/decorators/authorize.js +1 -1
  21. package/src/decorators/index.js +6 -6
  22. package/src/decorators/parameters.js +1 -1
  23. package/src/decorators/returns.js +1 -1
  24. package/src/decorators/scope.js +1 -1
  25. package/src/decorators/transacted.js +1 -1
  26. package/src/errors/AssetError.js +1 -1
  27. package/src/errors/AuthenticationError.js +1 -1
  28. package/src/errors/AuthorizationError.js +1 -1
  29. package/src/errors/ControllerError.js +1 -1
  30. package/src/errors/DatabaseError.js +12 -3
  31. package/src/errors/GraphError.js +1 -1
  32. package/src/errors/ModelError.js +1 -1
  33. package/src/errors/NotFoundError.js +1 -1
  34. package/src/errors/NotImplementedError.js +1 -1
  35. package/src/errors/QueryBuilderError.js +1 -1
  36. package/src/errors/RelationError.js +1 -1
  37. package/src/errors/ValidationError.js +1 -1
  38. package/src/errors/WrappedError.js +1 -1
  39. package/src/errors/index.js +14 -14
  40. package/src/graph/DitoGraphProcessor.js +2 -1
  41. package/src/graph/graph.js +1 -1
  42. package/src/graph/index.js +3 -3
  43. package/src/index.js +11 -9
  44. package/src/lib/index.js +2 -2
  45. package/src/middleware/createTransaction.js +1 -1
  46. package/src/middleware/findRoute.js +3 -2
  47. package/src/middleware/handleConnectMiddleware.js +88 -0
  48. package/src/middleware/handleError.js +1 -1
  49. package/src/middleware/handleRoute.js +3 -3
  50. package/src/middleware/index.js +8 -7
  51. package/src/mixins/AssetMixin.js +1 -1
  52. package/src/mixins/UserMixin.js +1 -1
  53. package/src/mixins/index.js +4 -4
  54. package/src/models/AssetModel.js +2 -2
  55. package/src/models/Model.js +16 -12
  56. package/src/models/RelationAccessor.js +1 -1
  57. package/src/models/SessionModel.js +2 -2
  58. package/src/models/TimeStampedModel.js +2 -2
  59. package/src/models/UserModel.js +2 -2
  60. package/src/models/definitions/assets.js +1 -1
  61. package/src/models/definitions/filters.js +57 -44
  62. package/src/models/definitions/hooks.js +1 -1
  63. package/src/models/definitions/index.js +9 -9
  64. package/src/models/definitions/modifiers.js +1 -1
  65. package/src/models/definitions/options.js +1 -1
  66. package/src/models/definitions/properties.js +2 -2
  67. package/src/models/definitions/relations.js +1 -1
  68. package/src/models/definitions/schema.js +1 -1
  69. package/src/models/definitions/scopes.js +2 -2
  70. package/src/models/index.js +5 -5
  71. package/src/query/QueryBuilder.js +5 -5
  72. package/src/query/QueryFilters.js +50 -50
  73. package/src/query/QueryParameters.js +2 -2
  74. package/src/query/index.js +3 -3
  75. package/src/schema/formats/index.js +2 -2
  76. package/src/schema/index.js +4 -4
  77. package/src/schema/keywords/_relate.js +1 -1
  78. package/src/schema/keywords/index.js +12 -12
  79. package/src/schema/properties.test.js +1 -1
  80. package/src/schema/relations.js +1 -1
  81. package/src/schema/relations.test.js +2 -2
  82. package/src/services/index.js +1 -1
  83. package/src/storage/DiskStorage.js +1 -1
  84. package/src/storage/S3Storage.js +4 -4
  85. package/src/storage/Storage.js +1 -1
  86. package/src/storage/index.js +4 -4
  87. package/src/utils/function.test.js +1 -1
  88. package/src/utils/handler.js +17 -0
  89. package/src/utils/index.js +8 -7
  90. package/src/utils/object.test.js +1 -1
  91. package/lib/app/Application.js +0 -961
  92. package/lib/app/SessionStore.js +0 -40
  93. package/lib/app/Validator.js +0 -355
  94. package/lib/app/index.js +0 -26
  95. package/lib/cli/console.js +0 -175
  96. package/lib/cli/db/createMigration.js +0 -237
  97. package/lib/cli/db/index.js +0 -66
  98. package/lib/cli/db/listAssetConfig.js +0 -16
  99. package/lib/cli/db/migrate.js +0 -15
  100. package/lib/cli/db/reset.js +0 -27
  101. package/lib/cli/db/rollback.js +0 -15
  102. package/lib/cli/db/seed.js +0 -104
  103. package/lib/cli/db/unlock.js +0 -15
  104. package/lib/cli/index.js +0 -90
  105. package/lib/controllers/AdminController.js +0 -258
  106. package/lib/controllers/CollectionController.js +0 -263
  107. package/lib/controllers/Controller.js +0 -462
  108. package/lib/controllers/ControllerAction.js +0 -276
  109. package/lib/controllers/MemberAction.js +0 -22
  110. package/lib/controllers/ModelController.js +0 -64
  111. package/lib/controllers/RelationController.js +0 -82
  112. package/lib/controllers/UserController.js +0 -98
  113. package/lib/controllers/index.js +0 -50
  114. package/lib/decorators/action.js +0 -14
  115. package/lib/decorators/authorize.js +0 -13
  116. package/lib/decorators/index.js +0 -58
  117. package/lib/decorators/parameters.js +0 -35
  118. package/lib/decorators/returns.js +0 -26
  119. package/lib/decorators/scope.js +0 -14
  120. package/lib/decorators/transacted.js +0 -12
  121. package/lib/errors/AssetError.js +0 -19
  122. package/lib/errors/AuthenticationError.js +0 -19
  123. package/lib/errors/AuthorizationError.js +0 -19
  124. package/lib/errors/ControllerError.js +0 -24
  125. package/lib/errors/DatabaseError.js +0 -27
  126. package/lib/errors/GraphError.js +0 -19
  127. package/lib/errors/ModelError.js +0 -24
  128. package/lib/errors/NotFoundError.js +0 -19
  129. package/lib/errors/NotImplementedError.js +0 -19
  130. package/lib/errors/QueryBuilderError.js +0 -19
  131. package/lib/errors/RelationError.js +0 -36
  132. package/lib/errors/ResponseError.js +0 -46
  133. package/lib/errors/ValidationError.js +0 -19
  134. package/lib/errors/WrappedError.js +0 -24
  135. package/lib/errors/index.js +0 -122
  136. package/lib/graph/DitoGraphProcessor.js +0 -185
  137. package/lib/graph/expression.js +0 -76
  138. package/lib/graph/graph.js +0 -300
  139. package/lib/graph/index.js +0 -34
  140. package/lib/index.js +0 -82
  141. package/lib/lib/EventEmitter.js +0 -76
  142. package/lib/lib/KnexHelper.js +0 -40
  143. package/lib/lib/index.js +0 -26
  144. package/lib/middleware/attachLogger.js +0 -16
  145. package/lib/middleware/createTransaction.js +0 -36
  146. package/lib/middleware/findRoute.js +0 -35
  147. package/lib/middleware/handleError.js +0 -26
  148. package/lib/middleware/handleRoute.js +0 -29
  149. package/lib/middleware/handleUser.js +0 -36
  150. package/lib/middleware/index.js +0 -66
  151. package/lib/middleware/logRequests.js +0 -122
  152. package/lib/mixins/AssetMixin.js +0 -81
  153. package/lib/mixins/SessionMixin.js +0 -22
  154. package/lib/mixins/TimeStampedMixin.js +0 -47
  155. package/lib/mixins/UserMixin.js +0 -151
  156. package/lib/mixins/index.js +0 -42
  157. package/lib/models/AssetModel.js +0 -12
  158. package/lib/models/Model.js +0 -953
  159. package/lib/models/RelationAccessor.js +0 -41
  160. package/lib/models/SessionModel.js +0 -12
  161. package/lib/models/TimeStampedModel.js +0 -12
  162. package/lib/models/UserModel.js +0 -12
  163. package/lib/models/definitions/assets.js +0 -11
  164. package/lib/models/definitions/filters.js +0 -101
  165. package/lib/models/definitions/hooks.js +0 -11
  166. package/lib/models/definitions/index.js +0 -38
  167. package/lib/models/definitions/modifiers.js +0 -11
  168. package/lib/models/definitions/options.js +0 -11
  169. package/lib/models/definitions/properties.js +0 -87
  170. package/lib/models/definitions/relations.js +0 -11
  171. package/lib/models/definitions/schema.js +0 -11
  172. package/lib/models/definitions/scopes.js +0 -51
  173. package/lib/models/index.js +0 -50
  174. package/lib/query/QueryBuilder.js +0 -745
  175. package/lib/query/QueryFilters.js +0 -82
  176. package/lib/query/QueryParameters.js +0 -77
  177. package/lib/query/Registry.js +0 -40
  178. package/lib/query/index.js +0 -34
  179. package/lib/schema/formats/_empty.js +0 -10
  180. package/lib/schema/formats/_required.js +0 -10
  181. package/lib/schema/formats/index.js +0 -26
  182. package/lib/schema/index.js +0 -49
  183. package/lib/schema/keywords/_computed.js +0 -11
  184. package/lib/schema/keywords/_foreign.js +0 -11
  185. package/lib/schema/keywords/_hidden.js +0 -11
  186. package/lib/schema/keywords/_index.js +0 -11
  187. package/lib/schema/keywords/_instanceof.js +0 -49
  188. package/lib/schema/keywords/_primary.js +0 -11
  189. package/lib/schema/keywords/_range.js +0 -26
  190. package/lib/schema/keywords/_relate.js +0 -19
  191. package/lib/schema/keywords/_specificType.js +0 -11
  192. package/lib/schema/keywords/_unique.js +0 -11
  193. package/lib/schema/keywords/_unsigned.js +0 -11
  194. package/lib/schema/keywords/_validate.js +0 -80
  195. package/lib/schema/keywords/index.js +0 -106
  196. package/lib/schema/properties.js +0 -227
  197. package/lib/schema/properties.test.js +0 -573
  198. package/lib/schema/relations.js +0 -274
  199. package/lib/schema/relations.test.js +0 -155
  200. package/lib/services/Service.js +0 -34
  201. package/lib/services/index.js +0 -18
  202. package/lib/storage/AssetFile.js +0 -97
  203. package/lib/storage/DiskStorage.js +0 -125
  204. package/lib/storage/S3Storage.js +0 -171
  205. package/lib/storage/Storage.js +0 -209
  206. package/lib/storage/index.js +0 -34
  207. package/lib/utils/decorator.js +0 -16
  208. package/lib/utils/deprecate.js +0 -46
  209. package/lib/utils/emitter.js +0 -13
  210. package/lib/utils/function.js +0 -14
  211. package/lib/utils/function.test.js +0 -49
  212. package/lib/utils/index.js +0 -66
  213. package/lib/utils/json.js +0 -9
  214. package/lib/utils/object.js +0 -92
  215. package/lib/utils/object.test.js +0 -65
  216. package/lib/utils/scope.js +0 -14
@@ -1,953 +0,0 @@
1
- "use strict";
2
-
3
- exports.__esModule = true;
4
- exports.Model = 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.reduce.js");
13
-
14
- require("core-js/modules/esnext.iterator.reduce.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.weak-map.delete-all.js");
21
-
22
- require("core-js/modules/esnext.weak-map.emplace.js");
23
-
24
- var _objection = _interopRequireDefault(require("objection"));
25
-
26
- var _query = require("../query");
27
-
28
- var _lib = require("../lib");
29
-
30
- var _schema = require("../schema");
31
-
32
- var _graph = require("../graph");
33
-
34
- var _utils = require("../utils");
35
-
36
- var _errors = require("../errors");
37
-
38
- var _utils2 = require("@ditojs/utils");
39
-
40
- var _RelationAccessor = _interopRequireDefault(require("./RelationAccessor"));
41
-
42
- var _definitions = _interopRequireDefault(require("./definitions"));
43
-
44
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
45
-
46
- class Model extends _objection.default.Model {
47
- constructor(json) {
48
- super();
49
-
50
- if (json) {
51
- this.$setJson(json);
52
- }
53
- }
54
-
55
- static setup(knex) {
56
- this.knex(knex);
57
-
58
- try {
59
- for (const relation of Object.values(this.getRelations())) {
60
- this.setupRelation(relation);
61
- }
62
- } catch (error) {
63
- throw error instanceof _errors.RelationError ? error : new _errors.RelationError(error);
64
- }
65
-
66
- this.referenceValidator = null;
67
- }
68
-
69
- static setupRelation(relation) {
70
- relation.relatedModelClass.getRelatedRelations().push(relation);
71
- const accessor = `$${relation.name}`;
72
-
73
- if (accessor in this.prototype) {
74
- throw new _errors.RelationError(`Model '${this.name}' already defines a property with name ` + `'${accessor}' that clashes with the relation accessor.`);
75
- }
76
-
77
- const defineAccessor = (target, isClass) => {
78
- Object.defineProperty(target, accessor, {
79
- get() {
80
- const value = new _RelationAccessor.default(relation, isClass ? this : null, isClass ? null : this);
81
- Object.defineProperty(this, accessor, {
82
- value,
83
- configurable: true,
84
- enumerable: false
85
- });
86
- return value;
87
- },
88
-
89
- configurable: true,
90
- enumerable: false
91
- });
92
- };
93
-
94
- defineAccessor(this, true);
95
- defineAccessor(this.prototype, false);
96
- }
97
-
98
- static initialize() {
99
- const {
100
- hooks,
101
- assets
102
- } = this.definition;
103
-
104
- this._setupEmitter(hooks);
105
-
106
- if (assets) {
107
- this._setupAssetsEvents(assets);
108
- }
109
- }
110
-
111
- $initialize() {}
112
-
113
- get $app() {
114
- return this.constructor.app;
115
- }
116
-
117
- $is(model) {
118
- return (model == null ? void 0 : model.constructor) === this.constructor && (model == null ? void 0 : model.id) === this.id;
119
- }
120
-
121
- $has(...properties) {
122
- for (const property of properties) {
123
- if (!(property in this)) return false;
124
- }
125
-
126
- return true;
127
- }
128
-
129
- $update(properties, trx) {
130
- return this.$query(trx).update(properties).runAfter((result, query) => query.has('update') ? this.$set(result) : result);
131
- }
132
-
133
- $patch(properties, trx) {
134
- return this.$query(trx).patch(properties).runAfter((result, query) => query.has('patch') ? this.$set(result) : result);
135
- }
136
-
137
- $transaction(trx, handler) {
138
- return this.constructor.transaction(trx, handler);
139
- }
140
-
141
- static transaction(trx, handler) {
142
- if (!handler) {
143
- handler = trx;
144
- trx = null;
145
- }
146
-
147
- if (handler) {
148
- return trx ? handler(trx) : this.knex().transaction(handler);
149
- } else {
150
- return super.transaction();
151
- }
152
- }
153
-
154
- $validate(json, options = {}) {
155
- if (options.skipValidation) {
156
- return json;
157
- }
158
-
159
- if (!options.graph && !options.async) {
160
- return super.$validate(json, options);
161
- }
162
-
163
- json = json || this;
164
- const inputJson = json;
165
- const shallow = json.$isObjectionModel && !options.graph;
166
-
167
- if (shallow) {
168
- json = json.clone({
169
- shallow: true
170
- });
171
- options = { ...options,
172
- mutable: true
173
- };
174
- }
175
-
176
- const validator = this.constructor.getValidator();
177
- const args = {
178
- options,
179
- model: this,
180
- json,
181
- ctx: Object.create(null)
182
- };
183
- validator.beforeValidate(args);
184
- const result = validator.validate(args);
185
-
186
- const handleResult = result => {
187
- validator.afterValidate(args);
188
- return shallow ? inputJson.$set(result) : result;
189
- };
190
-
191
- return (0, _utils2.isPromise)(result) ? result.then(handleResult) : handleResult(result);
192
- }
193
-
194
- async $validateGraph(options = {}) {
195
- await this.$validate(null, { ...options,
196
- graph: true,
197
- async: true
198
- });
199
- return this;
200
- }
201
-
202
- static fromJson(json, options = {}) {
203
- if (options.async && !options.skipValidation) {
204
- const model = new this();
205
- return model.$validate(json, options).then(json => model.$setJson(json, { ...options,
206
- skipValidation: true
207
- }));
208
- }
209
-
210
- return super.fromJson(json, options);
211
- }
212
-
213
- static query(trx) {
214
- return super.query(trx).onError(err => {
215
- err = err instanceof _errors.ResponseError ? err : err instanceof _objection.default.DBError ? this.app.createDatabaseError(err) : new _errors.WrappedError(err);
216
- return Promise.reject(err);
217
- });
218
- }
219
-
220
- static async count(...args) {
221
- const {
222
- count
223
- } = (await this.query().count(...args).first()) || {};
224
- return +count || 0;
225
- }
226
-
227
- static get tableName() {
228
- return this.name.match(/^(.*?)(?:Model|)$/)[1];
229
- }
230
-
231
- static get idColumn() {
232
- const {
233
- properties
234
- } = this;
235
- const ids = [];
236
-
237
- for (const [name, property] of Object.entries(properties || {})) {
238
- if (property != null && property.primary) {
239
- ids.push(name);
240
- }
241
- }
242
-
243
- const {
244
- length
245
- } = ids;
246
- return length > 1 ? ids : length > 0 ? ids[0] : super.idColumn;
247
- }
248
-
249
- static getReference(modelOrId, includeProperties) {
250
- const ref = new this();
251
- const idProperties = this.getIdPropertyArray();
252
-
253
- if ((0, _utils2.isObject)(modelOrId)) {
254
- const addProperty = key => {
255
- const value = modelOrId[key];
256
-
257
- if (value !== undefined) {
258
- ref[key] = value;
259
- }
260
- };
261
-
262
- addProperty(this.uidRefProp);
263
- idProperties.forEach(addProperty);
264
- includeProperties == null ? void 0 : includeProperties.forEach(addProperty);
265
- } else {
266
- const ids = (0, _utils2.asArray)(modelOrId);
267
-
268
- if (ids.length !== idProperties.length) {
269
- throw new _errors.ModelError(this, `Invalid amount of id values provided for reference: Unable to map ${(0, _utils.formatJson)(modelOrId, false)} to ${(0, _utils.formatJson)(idProperties, false)}.`);
270
- }
271
-
272
- idProperties.forEach((key, index) => {
273
- ref[key] = ids[index];
274
- });
275
- }
276
-
277
- return ref;
278
- }
279
-
280
- static isReference(obj) {
281
- let validator = this.referenceValidator;
282
-
283
- if (!validator) {
284
- validator = this.referenceValidator = this.app.compileValidator({
285
- oneOf: [{
286
- type: 'object',
287
- properties: this.getIdPropertyArray().reduce((idProperties, idProperty) => {
288
- idProperties[idProperty] = {
289
- type: this.definition.properties[idProperty].type
290
- };
291
- return idProperties;
292
- }, {}),
293
- additionalProperties: false
294
- }, {
295
- type: 'object',
296
- properties: {
297
- [this.uidRefProp]: {
298
- type: 'string'
299
- }
300
- },
301
- additionalProperties: false
302
- }]
303
- }, {
304
- throw: false
305
- });
306
- }
307
-
308
- return validator(obj);
309
- }
310
-
311
- static getScope(name) {
312
- return this.definition.scopes[name];
313
- }
314
-
315
- static hasScope(name) {
316
- return !!this.getScope(name);
317
- }
318
-
319
- static getModifiers() {
320
- return this.definition.modifiers;
321
- }
322
-
323
- static get relationMappings() {
324
- return this._getCached('relationMappings', () => (0, _schema.convertRelations)(this, this.definition.relations, this.app.models), {});
325
- }
326
-
327
- static get jsonSchema() {
328
- return this._getCached('jsonSchema', () => {
329
- const schema = (0, _schema.convertSchema)(this.definition.properties);
330
- (0, _schema.addRelationSchemas)(this, schema.properties);
331
- (0, _utils2.merge)(schema, this.definition.schema);
332
- return {
333
- $id: this.name,
334
- $schema: 'http://json-schema.org/draft-07/schema',
335
- ...schema
336
- };
337
- }, {});
338
- }
339
-
340
- static get virtualAttributes() {
341
- return this.computedAttributes;
342
- }
343
-
344
- static get jsonAttributes() {
345
- return this._getCached('jsonSchema:jsonAttributes', () => this.getAttributes(({
346
- type,
347
- specificType,
348
- computed
349
- }) => !computed && !specificType && (type === 'object' || type === 'array')), []);
350
- }
351
-
352
- static get booleanAttributes() {
353
- return this._getCached('jsonSchema:booleanAttributes', () => this.getAttributes(({
354
- type,
355
- computed
356
- }) => !computed && type === 'boolean'), []);
357
- }
358
-
359
- static get dateAttributes() {
360
- return this._getCached('jsonSchema:dateAttributes', () => this.getAttributes(({
361
- type,
362
- computed
363
- }) => !computed && ['date', 'datetime', 'timestamp'].includes(type)), []);
364
- }
365
-
366
- static get computedAttributes() {
367
- return this._getCached('jsonSchema:computedAttributes', () => this.getAttributes(({
368
- computed
369
- }) => computed), []);
370
- }
371
-
372
- static get hiddenAttributes() {
373
- return this._getCached('jsonSchema:hiddenAttributes', () => this.getAttributes(({
374
- hidden
375
- }) => hidden), []);
376
- }
377
-
378
- static getAttributes(filter) {
379
- const attributes = [];
380
- const {
381
- properties
382
- } = this.definition;
383
-
384
- for (const [name, property] of Object.entries(properties)) {
385
- if (filter(property)) {
386
- attributes.push(name);
387
- }
388
- }
389
-
390
- return attributes;
391
- }
392
-
393
- static _getCached(identifier, calculate, empty = {}) {
394
- var _entry, _entry2;
395
-
396
- let cache = getMeta(this, 'cache', {});
397
- let entry;
398
-
399
- for (const part of identifier.split(':')) {
400
- entry = cache[part] = cache[part] || {
401
- cache: {},
402
- value: undefined
403
- };
404
- cache = entry.cache;
405
- }
406
-
407
- if (((_entry = entry) == null ? void 0 : _entry.value) === undefined) {
408
- entry.value = empty;
409
- entry.value = calculate();
410
- entry.cache = {};
411
- }
412
-
413
- return (_entry2 = entry) == null ? void 0 : _entry2.value;
414
- }
415
-
416
- static getRelatedRelations() {
417
- return getMeta(this, 'relatedRelations', []);
418
- }
419
-
420
- static propertyNameToColumnName(propertyName) {
421
- return propertyName;
422
- }
423
-
424
- static columnNameToPropertyName(columnName) {
425
- return columnName;
426
- }
427
-
428
- $setJson(json, options) {
429
- options = options || {};
430
- const callInitialize = !options.patch && this.$initialize !== Model.prototype.$initialize && !this.constructor.isReference(json);
431
-
432
- if (!callInitialize || options.skipValidation) {
433
- super.$setJson(json, options);
434
-
435
- if (callInitialize) {
436
- this.$initialize();
437
- }
438
- } else {
439
- super.$setJson(json, { ...options,
440
- patch: true
441
- });
442
- this.$initialize();
443
- this.$validate(this, options);
444
- }
445
-
446
- return this;
447
- }
448
-
449
- $formatDatabaseJson(json) {
450
- const {
451
- constructor
452
- } = this;
453
-
454
- for (const key of constructor.dateAttributes) {
455
- const date = json[key];
456
-
457
- if (date != null && date.toISOString) {
458
- json[key] = date.toISOString();
459
- }
460
- }
461
-
462
- if (constructor.isSQLite()) {
463
- for (const key of constructor.booleanAttributes) {
464
- const bool = json[key];
465
-
466
- if (bool !== undefined) {
467
- json[key] = bool ? 1 : 0;
468
- }
469
- }
470
- }
471
-
472
- for (const key of constructor.computedAttributes) {
473
- delete json[key];
474
- }
475
-
476
- return super.$formatDatabaseJson(json);
477
- }
478
-
479
- $parseDatabaseJson(json) {
480
- const {
481
- constructor
482
- } = this;
483
- json = super.$parseDatabaseJson(json);
484
-
485
- if (constructor.isSQLite()) {
486
- for (const key of constructor.booleanAttributes) {
487
- const bool = json[key];
488
-
489
- if (bool !== undefined) {
490
- json[key] = !!bool;
491
- }
492
- }
493
- }
494
-
495
- return this.$parseJson(json);
496
- }
497
-
498
- $parseJson(json) {
499
- const {
500
- constructor
501
- } = this;
502
-
503
- for (const key of constructor.dateAttributes) {
504
- const date = json[key];
505
-
506
- if (date !== undefined) {
507
- json[key] = (0, _utils2.isString)(date) ? new Date(date) : date;
508
- }
509
- }
510
-
511
- const {
512
- assets
513
- } = constructor.definition;
514
-
515
- if (assets) {
516
- for (const dataPath in assets) {
517
- const storage = constructor.app.getStorage(assets[dataPath].storage);
518
- const data = (0, _utils2.getValueAtDataPath)(json, dataPath, () => null);
519
-
520
- if (data) {
521
- const convertToAssetFiles = data => {
522
- if (data) {
523
- if ((0, _utils2.isArray)(data)) {
524
- data.forEach(convertToAssetFiles);
525
- } else {
526
- storage.convertAssetFile(data);
527
- }
528
- }
529
- };
530
-
531
- convertToAssetFiles(data);
532
- }
533
- }
534
- }
535
-
536
- return json;
537
- }
538
-
539
- $formatJson(json) {
540
- const {
541
- constructor
542
- } = this;
543
-
544
- for (const key of constructor.computedAttributes) {
545
- if (!(key in json)) {
546
- const value = this[key];
547
-
548
- if (value !== undefined) {
549
- json[key] = value;
550
- }
551
- }
552
- }
553
-
554
- for (const key of constructor.hiddenAttributes) {
555
- delete json[key];
556
- }
557
-
558
- return json;
559
- }
560
-
561
- $filterGraph(modelGraph, expr) {
562
- return (0, _graph.filterGraph)(this.constructor, modelGraph, expr);
563
- }
564
-
565
- async $populateGraph(modelGraph, expr, trx) {
566
- return (0, _graph.populateGraph)(this.constructor, modelGraph, expr, trx);
567
- }
568
-
569
- static filterGraph(modelGraph, expr) {
570
- return (0, _graph.filterGraph)(this, modelGraph, expr);
571
- }
572
-
573
- static async populateGraph(modelGraph, expr, trx) {
574
- return (0, _graph.populateGraph)(this, modelGraph, expr, trx);
575
- }
576
-
577
- static getPropertyOrRelationAtDataPath(dataPath) {
578
- const parsedDataPath = (0, _utils2.parseDataPath)(dataPath);
579
- let index = 0;
580
-
581
- const getResult = (property = null, relation = null) => {
582
- const found = property || relation;
583
- const name = parsedDataPath[index];
584
- const next = index + 1;
585
- const dataPath = found ? (0, _utils2.normalizeDataPath)(parsedDataPath.slice(0, next)) : null;
586
- const nestedDataPath = found ? (0, _utils2.normalizeDataPath)(parsedDataPath.slice(next)) : null;
587
- const expression = found ? parsedDataPath.slice(0, relation ? next : index).join('.') + (property ? `(#${name})` : '') : null;
588
- return {
589
- property,
590
- relation,
591
- dataPath,
592
- nestedDataPath,
593
- name,
594
- expression,
595
- index
596
- };
597
- };
598
-
599
- const [firstToken, ...otherTokens] = parsedDataPath;
600
- const property = this.definition.properties[firstToken];
601
-
602
- if (property) {
603
- return getResult(property);
604
- } else {
605
- let relation = this.getRelations()[firstToken];
606
-
607
- if (relation) {
608
- let {
609
- relatedModelClass
610
- } = relation;
611
-
612
- for (const token of otherTokens) {
613
- index++;
614
- const property = relatedModelClass.definition.properties[token];
615
-
616
- if (property) {
617
- return getResult(property);
618
- } else if (token === '*') {
619
- if (relation.isOneToOne()) {
620
- return getResult();
621
- } else {
622
- continue;
623
- }
624
- } else {
625
- relation = relatedModelClass.getRelations()[token];
626
-
627
- if (relation) {
628
- relatedModelClass = relation.relatedModelClass;
629
- } else {
630
- return getResult();
631
- }
632
- }
633
- }
634
-
635
- if (relation) {
636
- return getResult(null, relation);
637
- }
638
- }
639
- }
640
-
641
- return getResult();
642
- }
643
-
644
- static relatedQuery(relationName, trx) {
645
- return super.relatedQuery(relationName, trx).alias(relationName);
646
- }
647
-
648
- static modifierNotFound(query, modifier) {
649
- if ((0, _utils2.isString)(modifier)) {
650
- if (query.modelClass().hasScope(modifier)) {
651
- return query.applyScope(modifier);
652
- }
653
-
654
- switch (modifier[0]) {
655
- case '^':
656
- return query.applyScope(modifier);
657
-
658
- case '-':
659
- return query.ignoreScope(modifier.slice(1));
660
-
661
- case '#':
662
- return query.select(modifier.slice(1));
663
- }
664
- }
665
-
666
- super.modifierNotFound(query, modifier);
667
- }
668
-
669
- static createNotFoundError(ctx, error) {
670
- return new _errors.NotFoundError(error || (ctx.byId ? `'${this.name}' model with id ${ctx.byId} not found` : `'${this.name}' model not found`));
671
- }
672
-
673
- static createValidator() {
674
- return this.app.validator;
675
- }
676
-
677
- static createValidationError({
678
- type,
679
- message,
680
- errors,
681
- options,
682
- json
683
- }) {
684
- switch (type) {
685
- case 'ModelValidation':
686
- return this.app.createValidationError({
687
- type,
688
- message: message || `The provided data for the ${this.name} model is not valid`,
689
- errors,
690
- options,
691
- json
692
- });
693
-
694
- case 'RelationExpression':
695
- case 'UnallowedRelation':
696
- return new _errors.RelationError({
697
- type,
698
- message,
699
- errors
700
- });
701
-
702
- case 'InvalidGraph':
703
- return new _errors.GraphError({
704
- type,
705
- message,
706
- errors
707
- });
708
-
709
- default:
710
- return new _errors.ResponseError({
711
- type,
712
- message,
713
- errors
714
- });
715
- }
716
- }
717
-
718
- static get definition() {
719
- return getMeta(this, 'definition', () => {
720
- const definition = {};
721
-
722
- const setDefinition = (name, property) => {
723
- Object.defineProperty(definition, name, { ...property,
724
- enumerable: true
725
- });
726
- };
727
-
728
- const getDefinition = name => {
729
- let modelClass = this;
730
- const values = [];
731
-
732
- while (modelClass !== _objection.default.Model) {
733
- if (name in modelClass) {
734
- const desc = Object.getOwnPropertyDescriptor(modelClass, name);
735
-
736
- if (desc) {
737
- var _desc$get;
738
-
739
- const value = ((_desc$get = desc.get) == null ? void 0 : _desc$get.call(this)) || desc.value;
740
-
741
- if (value) {
742
- values.push(value);
743
- }
744
- }
745
- }
746
-
747
- modelClass = Object.getPrototypeOf(modelClass);
748
- }
749
-
750
- setDefinition(name, {
751
- configurable: true,
752
- value: {}
753
- });
754
-
755
- try {
756
- const merged = _definitions.default[name].call(this, values);
757
-
758
- setDefinition(name, {
759
- configurable: false,
760
- value: merged
761
- });
762
- return merged;
763
- } catch (error) {
764
- throw new _errors.ModelError(this, error.message);
765
- }
766
- };
767
-
768
- for (const name in _definitions.default) {
769
- setDefinition(name, {
770
- configurable: true,
771
- get: () => getDefinition(name)
772
- });
773
- }
774
-
775
- return definition;
776
- });
777
- }
778
-
779
- $emit(event, ...args) {
780
- return this.constructor.emit(event, this, ...args);
781
- }
782
-
783
- static beforeFind(args) {
784
- return this._emitStaticHook('before:find', args);
785
- }
786
-
787
- static afterFind(args) {
788
- return this._emitStaticHook('after:find', args);
789
- }
790
-
791
- static beforeInsert(args) {
792
- return this._emitStaticHook('before:insert', args);
793
- }
794
-
795
- static afterInsert(args) {
796
- return this._emitStaticHook('after:insert', args);
797
- }
798
-
799
- static beforeUpdate(args) {
800
- return this._emitStaticHook('before:update', args);
801
- }
802
-
803
- static afterUpdate(args) {
804
- return this._emitStaticHook('after:update', args);
805
- }
806
-
807
- static beforeDelete(args) {
808
- return this._emitStaticHook('before:delete', args);
809
- }
810
-
811
- static afterDelete(args) {
812
- return this._emitStaticHook('after:delete', args);
813
- }
814
-
815
- static async _emitStaticHook(event, originalArgs) {
816
- const listeners = this.listeners(event);
817
-
818
- if (listeners.length > 0) {
819
- let {
820
- result
821
- } = originalArgs;
822
- const args = Object.create(originalArgs, {
823
- type: {
824
- value: event
825
- },
826
- result: {
827
- get() {
828
- return result;
829
- }
830
-
831
- }
832
- });
833
-
834
- for (const listener of listeners) {
835
- const res = await listener.call(this, args);
836
-
837
- if (res !== undefined) {
838
- result = res;
839
- }
840
- }
841
-
842
- if (result !== originalArgs.result) {
843
- return result;
844
- }
845
- }
846
- }
847
-
848
- static _setupAssetsEvents(assets) {
849
- const assetDataPaths = Object.keys(assets);
850
- this.on(['before:insert', 'before:update', 'before:delete'], async ({
851
- type,
852
- transaction,
853
- inputItems,
854
- asFindQuery
855
- }) => {
856
- const afterItems = type === 'before:delete' ? [] : inputItems;
857
- const dataPaths = afterItems.length > 0 ? assetDataPaths.filter(path => getValueAtAssetDataPath(afterItems[0], path) !== undefined) : assetDataPaths;
858
-
859
- if (dataPaths.length === 0) {
860
- return;
861
- }
862
-
863
- const beforeItems = type === 'before:insert' ? [] : await loadAssetDataPaths(asFindQuery(), dataPaths);
864
- const beforeFilesPerDataPath = getFilesPerAssetDataPath(beforeItems, dataPaths);
865
- const afterFilesPerDataPath = getFilesPerAssetDataPath(afterItems, dataPaths);
866
- const importedFiles = [];
867
- const modifiedFiles = [];
868
-
869
- if (transaction.rollback) {
870
- transaction.setMaxListeners(0);
871
- transaction.on('rollback', async error => {
872
- if (importedFiles.length > 0) {
873
- console.info(`Received '${error}', removing imported files again: ${importedFiles.map(file => `'${file.name}'`)}`);
874
- await Promise.all(importedFiles.map(file => file.storage.removeFile(file)));
875
- }
876
-
877
- if (modifiedFiles.length > 0) {
878
- console.info(`Unable to restore these already modified files: ${modifiedFiles.map(file => `'${file.name}'`)}`);
879
- }
880
- });
881
- }
882
-
883
- for (const dataPath of dataPaths) {
884
- const storage = this.app.getStorage(assets[dataPath].storage);
885
- const beforeFiles = beforeFilesPerDataPath[dataPath] || [];
886
- const afterFiles = afterFilesPerDataPath[dataPath] || [];
887
- const beforeByKey = mapFilesByKey(beforeFiles);
888
- const afterByKey = mapFilesByKey(afterFiles);
889
- const removedFiles = beforeFiles.filter(file => !afterByKey[file.key]);
890
- const addedFiles = afterFiles.filter(file => !beforeByKey[file.key]);
891
- const modifiedFiles = afterFiles.filter(file => file.data && beforeByKey[file.key]);
892
- importedFiles.push(...(await this.app.handleAdddedAndRemovedAssets(storage, addedFiles, removedFiles, transaction)));
893
- modifiedFiles.push(...(await this.app.handleModifiedAssets(storage, modifiedFiles, transaction)));
894
- }
895
- });
896
- }
897
-
898
- }
899
-
900
- exports.Model = Model;
901
- Model.QueryBuilder = _query.QueryBuilder;
902
- Model.cloneObjectAttributes = false;
903
- Model.pickJsonSchemaProperties = true;
904
- Model.useLimitInFirst = true;
905
-
906
- _lib.EventEmitter.mixin(Model);
907
-
908
- _lib.KnexHelper.mixin(Model);
909
-
910
- _query.QueryBuilder.mixin(Model);
911
-
912
- const metaMap = new WeakMap();
913
-
914
- function getMeta(modelClass, key, value) {
915
- let meta = metaMap.get(modelClass);
916
-
917
- if (!meta) {
918
- metaMap.set(modelClass, meta = {});
919
- }
920
-
921
- if (!(key in meta)) {
922
- meta[key] = (0, _utils2.isFunction)(value) ? value() : value;
923
- }
924
-
925
- return meta[key];
926
- }
927
-
928
- function loadAssetDataPaths(query, dataPaths) {
929
- return dataPaths.reduce((query, dataPath) => query.loadDataPath(dataPath), query);
930
- }
931
-
932
- function getValueAtAssetDataPath(item, path) {
933
- return (0, _utils2.getValueAtDataPath)(item, path, () => undefined);
934
- }
935
-
936
- function getFilesPerAssetDataPath(items, dataPaths) {
937
- return dataPaths.reduce((allFiles, dataPath) => {
938
- allFiles[dataPath] = (0, _utils2.asArray)(items).reduce((files, item) => {
939
- const data = (0, _utils2.asArray)(getValueAtAssetDataPath(item, dataPath));
940
- files.push(...(0, _utils2.flatten)(data).filter(file => !!file));
941
- return files;
942
- }, []);
943
- return allFiles;
944
- }, {});
945
- }
946
-
947
- function mapFilesByKey(files) {
948
- return files.reduce((map, file) => {
949
- map[file.key] = file;
950
- return map;
951
- }, {});
952
- }
953
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/models/Model.js"],"names":["Model","objection","constructor","json","$setJson","setup","knex","relation","Object","values","getRelations","setupRelation","error","RelationError","referenceValidator","relatedModelClass","getRelatedRelations","push","accessor","name","prototype","defineAccessor","target","isClass","defineProperty","get","value","RelationAccessor","configurable","enumerable","initialize","hooks","assets","definition","_setupEmitter","_setupAssetsEvents","$initialize","$app","app","$is","model","id","$has","properties","property","$update","trx","$query","update","runAfter","result","query","has","$set","$patch","patch","$transaction","handler","transaction","$validate","options","skipValidation","graph","async","inputJson","shallow","$isObjectionModel","clone","mutable","validator","getValidator","args","ctx","create","beforeValidate","validate","handleResult","afterValidate","then","$validateGraph","fromJson","onError","err","ResponseError","DBError","createDatabaseError","WrappedError","Promise","reject","count","first","tableName","match","idColumn","ids","entries","primary","length","getReference","modelOrId","includeProperties","ref","idProperties","getIdPropertyArray","addProperty","key","undefined","uidRefProp","forEach","ModelError","index","isReference","obj","compileValidator","oneOf","type","reduce","idProperty","additionalProperties","throw","getScope","scopes","hasScope","getModifiers","modifiers","relationMappings","_getCached","relations","models","jsonSchema","schema","$id","$schema","virtualAttributes","computedAttributes","jsonAttributes","getAttributes","specificType","computed","booleanAttributes","dateAttributes","includes","hiddenAttributes","hidden","filter","attributes","identifier","calculate","empty","cache","getMeta","entry","part","split","propertyNameToColumnName","propertyName","columnNameToPropertyName","columnName","callInitialize","$formatDatabaseJson","date","toISOString","isSQLite","bool","$parseDatabaseJson","$parseJson","Date","dataPath","storage","getStorage","data","convertToAssetFiles","convertAssetFile","$formatJson","$filterGraph","modelGraph","expr","$populateGraph","filterGraph","populateGraph","getPropertyOrRelationAtDataPath","parsedDataPath","getResult","found","next","slice","nestedDataPath","expression","join","firstToken","otherTokens","token","isOneToOne","relatedQuery","relationName","alias","modifierNotFound","modifier","modelClass","applyScope","ignoreScope","select","createNotFoundError","NotFoundError","byId","createValidator","createValidationError","message","errors","GraphError","setDefinition","getDefinition","desc","getOwnPropertyDescriptor","call","getPrototypeOf","merged","definitions","$emit","event","emit","beforeFind","_emitStaticHook","afterFind","beforeInsert","afterInsert","beforeUpdate","afterUpdate","beforeDelete","afterDelete","originalArgs","listeners","listener","res","assetDataPaths","keys","on","inputItems","asFindQuery","afterItems","dataPaths","path","getValueAtAssetDataPath","beforeItems","loadAssetDataPaths","beforeFilesPerDataPath","getFilesPerAssetDataPath","afterFilesPerDataPath","importedFiles","modifiedFiles","rollback","setMaxListeners","console","info","map","file","all","removeFile","beforeFiles","afterFiles","beforeByKey","mapFilesByKey","afterByKey","removedFiles","addedFiles","handleAdddedAndRemovedAssets","handleModifiedAssets","QueryBuilder","cloneObjectAttributes","pickJsonSchemaProperties","useLimitInFirst","EventEmitter","mixin","KnexHelper","metaMap","WeakMap","meta","set","loadDataPath","item","items","allFiles","files"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAOA;;AAIA;;AACA;;;;AAEO,MAAMA,KAAN,SAAoBC,mBAAUD,KAA9B,CAAoC;AAGzCE,EAAAA,WAAW,CAACC,IAAD,EAAO;AAChB;;AACA,QAAIA,IAAJ,EAAU;AACR,WAAKC,QAAL,CAAcD,IAAd;AACD;AACF;;AAEW,SAALE,KAAK,CAACC,IAAD,EAAO;AACjB,SAAKA,IAAL,CAAUA,IAAV;;AACA,QAAI;AACF,WAAK,MAAMC,QAAX,IAAuBC,MAAM,CAACC,MAAP,CAAc,KAAKC,YAAL,EAAd,CAAvB,EAA2D;AACzD,aAAKC,aAAL,CAAmBJ,QAAnB;AACD;AACF,KAJD,CAIE,OAAOK,KAAP,EAAc;AACd,YAAMA,KAAK,YAAYC,qBAAjB,GAAiCD,KAAjC,GAAyC,IAAIC,qBAAJ,CAAkBD,KAAlB,CAA/C;AACD;;AACD,SAAKE,kBAAL,GAA0B,IAA1B;AACD;;AAEmB,SAAbH,aAAa,CAACJ,QAAD,EAAW;AAG7BA,IAAAA,QAAQ,CAACQ,iBAAT,CAA2BC,mBAA3B,GAAiDC,IAAjD,CAAsDV,QAAtD;AAKA,UAAMW,QAAQ,GAAI,IAAGX,QAAQ,CAACY,IAAK,EAAnC;;AACA,QAAID,QAAQ,IAAI,KAAKE,SAArB,EAAgC;AAC9B,YAAM,IAAIP,qBAAJ,CACH,UAAS,KAAKM,IAAK,yCAApB,GACC,IAAGD,QAAS,4CAFT,CAAN;AAGD;;AAKD,UAAMG,cAAc,GAAG,CAACC,MAAD,EAASC,OAAT,KAAqB;AAC1Cf,MAAAA,MAAM,CAACgB,cAAP,CAAsBF,MAAtB,EAA8BJ,QAA9B,EAAwC;AACtCO,QAAAA,GAAG,GAAG;AACJ,gBAAMC,KAAK,GAAG,IAAIC,yBAAJ,CACZpB,QADY,EAEZgB,OAAO,GAAG,IAAH,GAAU,IAFL,EAGZA,OAAO,GAAG,IAAH,GAAU,IAHL,CAAd;AAMAf,UAAAA,MAAM,CAACgB,cAAP,CAAsB,IAAtB,EAA4BN,QAA5B,EAAsC;AACpCQ,YAAAA,KADoC;AAEpCE,YAAAA,YAAY,EAAE,IAFsB;AAGpCC,YAAAA,UAAU,EAAE;AAHwB,WAAtC;AAKA,iBAAOH,KAAP;AACD,SAdqC;;AAetCE,QAAAA,YAAY,EAAE,IAfwB;AAgBtCC,QAAAA,UAAU,EAAE;AAhB0B,OAAxC;AAkBD,KAnBD;;AAqBAR,IAAAA,cAAc,CAAC,IAAD,EAAO,IAAP,CAAd;AACAA,IAAAA,cAAc,CAAC,KAAKD,SAAN,EAAiB,KAAjB,CAAd;AACD;;AAGgB,SAAVU,UAAU,GAAG;AAClB,UAAM;AAAEC,MAAAA,KAAF;AAASC,MAAAA;AAAT,QAAoB,KAAKC,UAA/B;;AACA,SAAKC,aAAL,CAAmBH,KAAnB;;AACA,QAAIC,MAAJ,EAAY;AACV,WAAKG,kBAAL,CAAwBH,MAAxB;AACD;AACF;;AAGDI,EAAAA,WAAW,GAAG,CACb;;AAEO,MAAJC,IAAI,GAAG;AACT,WAAO,KAAKnC,WAAL,CAAiBoC,GAAxB;AACD;;AAEDC,EAAAA,GAAG,CAACC,KAAD,EAAQ;AACT,WAAO,CAAAA,KAAK,QAAL,YAAAA,KAAK,CAAEtC,WAAP,MAAuB,KAAKA,WAA5B,IAA2C,CAAAsC,KAAK,QAAL,YAAAA,KAAK,CAAEC,EAAP,MAAc,KAAKA,EAArE;AACD;;AAEDC,EAAAA,IAAI,CAAC,GAAGC,UAAJ,EAAgB;AAClB,SAAK,MAAMC,QAAX,IAAuBD,UAAvB,EAAmC;AACjC,UAAI,EAAEC,QAAQ,IAAI,IAAd,CAAJ,EAAyB,OAAO,KAAP;AAC1B;;AACD,WAAO,IAAP;AACD;;AAEDC,EAAAA,OAAO,CAACF,UAAD,EAAaG,GAAb,EAAkB;AACvB,WAAO,KAAKC,MAAL,CAAYD,GAAZ,EACJE,MADI,CACGL,UADH,EAEJM,QAFI,CAEK,CAACC,MAAD,EAASC,KAAT,KAGRA,KAAK,CAACC,GAAN,CAAU,QAAV,IAAsB,KAAKC,IAAL,CAAUH,MAAV,CAAtB,GAA0CA,MALvC,CAAP;AAOD;;AAEDI,EAAAA,MAAM,CAACX,UAAD,EAAaG,GAAb,EAAkB;AACtB,WAAO,KAAKC,MAAL,CAAYD,GAAZ,EACJS,KADI,CACEZ,UADF,EAEJM,QAFI,CAEK,CAACC,MAAD,EAASC,KAAT,KAGRA,KAAK,CAACC,GAAN,CAAU,OAAV,IAAqB,KAAKC,IAAL,CAAUH,MAAV,CAArB,GAAyCA,MALtC,CAAP;AAOD;;AAGDM,EAAAA,YAAY,CAACV,GAAD,EAAMW,OAAN,EAAe;AACzB,WAAO,KAAKvD,WAAL,CAAiBwD,WAAjB,CAA6BZ,GAA7B,EAAkCW,OAAlC,CAAP;AACD;;AAGiB,SAAXC,WAAW,CAACZ,GAAD,EAAMW,OAAN,EAAe;AAE/B,QAAI,CAACA,OAAL,EAAc;AACZA,MAAAA,OAAO,GAAGX,GAAV;AACAA,MAAAA,GAAG,GAAG,IAAN;AACD;;AACD,QAAIW,OAAJ,EAAa;AAEX,aAAOX,GAAG,GACNW,OAAO,CAACX,GAAD,CADD,GAEN,KAAKxC,IAAL,GAAYoD,WAAZ,CAAwBD,OAAxB,CAFJ;AAGD,KALD,MAKO;AAEL,aAAO,MAAMC,WAAN,EAAP;AACD;AACF;;AAGDC,EAAAA,SAAS,CAACxD,IAAD,EAAOyD,OAAO,GAAG,EAAjB,EAAqB;AAC5B,QAAIA,OAAO,CAACC,cAAZ,EAA4B;AAC1B,aAAO1D,IAAP;AACD;;AACD,QAAI,CAACyD,OAAO,CAACE,KAAT,IAAkB,CAACF,OAAO,CAACG,KAA/B,EAAsC;AAGpC,aAAO,MAAMJ,SAAN,CAAgBxD,IAAhB,EAAsByD,OAAtB,CAAP;AACD;;AACDzD,IAAAA,IAAI,GAAGA,IAAI,IAAI,IAAf;AACA,UAAM6D,SAAS,GAAG7D,IAAlB;AACA,UAAM8D,OAAO,GAAG9D,IAAI,CAAC+D,iBAAL,IAA0B,CAACN,OAAO,CAACE,KAAnD;;AACA,QAAIG,OAAJ,EAAa;AAEX9D,MAAAA,IAAI,GAAGA,IAAI,CAACgE,KAAL,CAAW;AAAEF,QAAAA,OAAO,EAAE;AAAX,OAAX,CAAP;AAEAL,MAAAA,OAAO,GAAG,EAAE,GAAGA,OAAL;AAAcQ,QAAAA,OAAO,EAAE;AAAvB,OAAV;AACD;;AAED,UAAMC,SAAS,GAAG,KAAKnE,WAAL,CAAiBoE,YAAjB,EAAlB;AACA,UAAMC,IAAI,GAAG;AACXX,MAAAA,OADW;AAEXpB,MAAAA,KAAK,EAAE,IAFI;AAGXrC,MAAAA,IAHW;AAIXqE,MAAAA,GAAG,EAAEhE,MAAM,CAACiE,MAAP,CAAc,IAAd;AAJM,KAAb;AAOAJ,IAAAA,SAAS,CAACK,cAAV,CAAyBH,IAAzB;AACA,UAAMrB,MAAM,GAAGmB,SAAS,CAACM,QAAV,CAAmBJ,IAAnB,CAAf;;AACA,UAAMK,YAAY,GAAG1B,MAAM,IAAI;AAC7BmB,MAAAA,SAAS,CAACQ,aAAV,CAAwBN,IAAxB;AAEA,aAAON,OAAO,GAAGD,SAAS,CAACX,IAAV,CAAeH,MAAf,CAAH,GAA4BA,MAA1C;AACD,KAJD;;AAMA,WAAO,uBAAUA,MAAV,IACHA,MAAM,CAAC4B,IAAP,CAAYF,YAAZ,CADG,GAEHA,YAAY,CAAC1B,MAAD,CAFhB;AAGD;;AAEmB,QAAd6B,cAAc,CAACnB,OAAO,GAAG,EAAX,EAAe;AACjC,UAAM,KAAKD,SAAL,CAAe,IAAf,EAAqB,EACzB,GAAGC,OADsB;AAEzBE,MAAAA,KAAK,EAAE,IAFkB;AAIzBC,MAAAA,KAAK,EAAE;AAJkB,KAArB,CAAN;AAMA,WAAO,IAAP;AACD;;AAGc,SAARiB,QAAQ,CAAC7E,IAAD,EAAOyD,OAAO,GAAG,EAAjB,EAAqB;AAClC,QAAIA,OAAO,CAACG,KAAR,IAAiB,CAACH,OAAO,CAACC,cAA9B,EAA8C;AAE5C,YAAMrB,KAAK,GAAG,IAAI,IAAJ,EAAd;AACA,aAAOA,KAAK,CAACmB,SAAN,CAAgBxD,IAAhB,EAAsByD,OAAtB,EAA+BkB,IAA/B,CACL3E,IAAI,IAAIqC,KAAK,CAACpC,QAAN,CAAeD,IAAf,EAAqB,EAC3B,GAAGyD,OADwB;AAE3BC,QAAAA,cAAc,EAAE;AAFW,OAArB,CADH,CAAP;AAMD;;AAED,WAAO,MAAMmB,QAAN,CAAe7E,IAAf,EAAqByD,OAArB,CAAP;AACD;;AAGW,SAALT,KAAK,CAACL,GAAD,EAAM;AAChB,WAAO,MAAMK,KAAN,CAAYL,GAAZ,EAAiBmC,OAAjB,CAAyBC,GAAG,IAAI;AAErCA,MAAAA,GAAG,GAAGA,GAAG,YAAYC,qBAAf,GAA+BD,GAA/B,GACFA,GAAG,YAAYjF,mBAAUmF,OAAzB,GAAmC,KAAK9C,GAAL,CAAS+C,mBAAT,CAA6BH,GAA7B,CAAnC,GACA,IAAII,oBAAJ,CAAiBJ,GAAjB,CAFJ;AAGA,aAAOK,OAAO,CAACC,MAAR,CAAeN,GAAf,CAAP;AACD,KANM,CAAP;AAOD;;AAEiB,eAALO,KAAK,CAAC,GAAGlB,IAAJ,EAAU;AAC1B,UAAM;AAAEkB,MAAAA;AAAF,QAAY,OAAM,KAAKtC,KAAL,GAAasC,KAAb,CAAmB,GAAGlB,IAAtB,EAA4BmB,KAA5B,EAAN,KAA6C,EAA/D;AACA,WAAO,CAACD,KAAD,IAAU,CAAjB;AACD;;AAGmB,aAATE,SAAS,GAAG;AAErB,WAAO,KAAKxE,IAAL,CAAUyE,KAAV,CAAgB,mBAAhB,EAAqC,CAArC,CAAP;AACD;;AAGkB,aAARC,QAAQ,GAAG;AAIpB,UAAM;AAAElD,MAAAA;AAAF,QAAiB,IAAvB;AACA,UAAMmD,GAAG,GAAG,EAAZ;;AACA,SAAK,MAAM,CAAC3E,IAAD,EAAOyB,QAAP,CAAX,IAA+BpC,MAAM,CAACuF,OAAP,CAAepD,UAAU,IAAI,EAA7B,CAA/B,EAAiE;AAC/D,UAAIC,QAAJ,YAAIA,QAAQ,CAAEoD,OAAd,EAAuB;AACrBF,QAAAA,GAAG,CAAC7E,IAAJ,CAASE,IAAT;AACD;AACF;;AACD,UAAM;AAAE8E,MAAAA;AAAF,QAAaH,GAAnB;AACA,WAAOG,MAAM,GAAG,CAAT,GAAaH,GAAb,GAAmBG,MAAM,GAAG,CAAT,GAAaH,GAAG,CAAC,CAAD,CAAhB,GAAsB,MAAMD,QAAtD;AACD;;AAEkB,SAAZK,YAAY,CAACC,SAAD,EAAYC,iBAAZ,EAA+B;AAIhD,UAAMC,GAAG,GAAG,IAAI,IAAJ,EAAZ;AACA,UAAMC,YAAY,GAAG,KAAKC,kBAAL,EAArB;;AACA,QAAI,sBAASJ,SAAT,CAAJ,EAAyB;AACvB,YAAMK,WAAW,GAAGC,GAAG,IAAI;AACzB,cAAM/E,KAAK,GAAGyE,SAAS,CAACM,GAAD,CAAvB;;AACA,YAAI/E,KAAK,KAAKgF,SAAd,EAAyB;AACvBL,UAAAA,GAAG,CAACI,GAAD,CAAH,GAAW/E,KAAX;AACD;AACF,OALD;;AAOA8E,MAAAA,WAAW,CAAC,KAAKG,UAAN,CAAX;AACAL,MAAAA,YAAY,CAACM,OAAb,CAAqBJ,WAArB;AACAJ,MAAAA,iBAAiB,QAAjB,YAAAA,iBAAiB,CAAEQ,OAAnB,CAA2BJ,WAA3B;AACD,KAXD,MAWO;AAEL,YAAMV,GAAG,GAAG,qBAAQK,SAAR,CAAZ;;AACA,UAAIL,GAAG,CAACG,MAAJ,KAAeK,YAAY,CAACL,MAAhC,EAAwC;AACtC,cAAM,IAAIY,kBAAJ,CACJ,IADI,EAEH,qEACC,uBAAWV,SAAX,EAAsB,KAAtB,CACD,OACC,uBAAWG,YAAX,EAAyB,KAAzB,CACD,GANG,CAAN;AAQD;;AACDA,MAAAA,YAAY,CAACM,OAAb,CAAqB,CAACH,GAAD,EAAMK,KAAN,KAAgB;AACnCT,QAAAA,GAAG,CAACI,GAAD,CAAH,GAAWX,GAAG,CAACgB,KAAD,CAAd;AACD,OAFD;AAGD;;AACD,WAAOT,GAAP;AACD;;AAEiB,SAAXU,WAAW,CAACC,GAAD,EAAM;AACtB,QAAI3C,SAAS,GAAG,KAAKvD,kBAArB;;AACA,QAAI,CAACuD,SAAL,EAAgB;AAGdA,MAAAA,SAAS,GAAG,KAAKvD,kBAAL,GAA0B,KAAKwB,GAAL,CAAS2E,gBAAT,CACpC;AACEC,QAAAA,KAAK,EAAE,CACL;AACEC,UAAAA,IAAI,EAAE,QADR;AAGExE,UAAAA,UAAU,EAAE,KAAK4D,kBAAL,GAA0Ba,MAA1B,CACV,CAACd,YAAD,EAAee,UAAf,KAA8B;AAC5Bf,YAAAA,YAAY,CAACe,UAAD,CAAZ,GAA2B;AACzBF,cAAAA,IAAI,EAAE,KAAKlF,UAAL,CAAgBU,UAAhB,CAA2B0E,UAA3B,EAAuCF;AADpB,aAA3B;AAGA,mBAAOb,YAAP;AACD,WANS,EAOV,EAPU,CAHd;AAYEgB,UAAAA,oBAAoB,EAAE;AAZxB,SADK,EAeL;AACEH,UAAAA,IAAI,EAAE,QADR;AAEExE,UAAAA,UAAU,EAAE;AACV,aAAC,KAAKgE,UAAN,GAAmB;AACjBQ,cAAAA,IAAI,EAAE;AADW;AADT,WAFd;AAOEG,UAAAA,oBAAoB,EAAE;AAPxB,SAfK;AADT,OADoC,EA6BpC;AAAEC,QAAAA,KAAK,EAAE;AAAT,OA7BoC,CAAtC;AA+BD;;AACD,WAAOlD,SAAS,CAAC2C,GAAD,CAAhB;AACD;;AAEc,SAARQ,QAAQ,CAACrG,IAAD,EAAO;AACpB,WAAO,KAAKc,UAAL,CAAgBwF,MAAhB,CAAuBtG,IAAvB,CAAP;AACD;;AAEc,SAARuG,QAAQ,CAACvG,IAAD,EAAO;AACpB,WAAO,CAAC,CAAC,KAAKqG,QAAL,CAAcrG,IAAd,CAAT;AACD;;AAEkB,SAAZwG,YAAY,GAAG;AACpB,WAAO,KAAK1F,UAAL,CAAgB2F,SAAvB;AACD;;AAE0B,aAAhBC,gBAAgB,GAAG;AAC5B,WAAO,KAAKC,UAAL,CAAgB,kBAAhB,EAAoC,MACzC,8BAAiB,IAAjB,EAAuB,KAAK7F,UAAL,CAAgB8F,SAAvC,EAAkD,KAAKzF,GAAL,CAAS0F,MAA3D,CADK,EAEJ,EAFI,CAAP;AAGD;;AAEoB,aAAVC,UAAU,GAAG;AACtB,WAAO,KAAKH,UAAL,CAAgB,YAAhB,EAA8B,MAAM;AACzC,YAAMI,MAAM,GAAG,2BAAc,KAAKjG,UAAL,CAAgBU,UAA9B,CAAf;AACA,sCAAmB,IAAnB,EAAyBuF,MAAM,CAACvF,UAAhC;AAEA,yBAAMuF,MAAN,EAAc,KAAKjG,UAAL,CAAgBiG,MAA9B;AACA,aAAO;AACLC,QAAAA,GAAG,EAAE,KAAKhH,IADL;AAELiH,QAAAA,OAAO,EAAE,wCAFJ;AAGL,WAAGF;AAHE,OAAP;AAKD,KAVM,EAUJ,EAVI,CAAP;AAWD;;AAE2B,aAAjBG,iBAAiB,GAAG;AAG7B,WAAO,KAAKC,kBAAZ;AACD;;AAEwB,aAAdC,cAAc,GAAG;AAC1B,WAAO,KAAKT,UAAL,CAAgB,2BAAhB,EAA6C,MAClD,KAAKU,aAAL,CAAmB,CAAC;AAAErB,MAAAA,IAAF;AAAQsB,MAAAA,YAAR;AAAsBC,MAAAA;AAAtB,KAAD,KACjB,CAACA,QAAD,IAAa,CAACD,YAAd,KAA+BtB,IAAI,KAAK,QAAT,IAAqBA,IAAI,KAAK,OAA7D,CADF,CADK,EAGJ,EAHI,CAAP;AAID;;AAE2B,aAAjBwB,iBAAiB,GAAG;AAC7B,WAAO,KAAKb,UAAL,CAAgB,8BAAhB,EAAgD,MACrD,KAAKU,aAAL,CAAmB,CAAC;AAAErB,MAAAA,IAAF;AAAQuB,MAAAA;AAAR,KAAD,KACjB,CAACA,QAAD,IAAavB,IAAI,KAAK,SADxB,CADK,EAGJ,EAHI,CAAP;AAID;;AAEwB,aAAdyB,cAAc,GAAG;AAC1B,WAAO,KAAKd,UAAL,CAAgB,2BAAhB,EAA6C,MAClD,KAAKU,aAAL,CAAmB,CAAC;AAAErB,MAAAA,IAAF;AAAQuB,MAAAA;AAAR,KAAD,KACjB,CAACA,QAAD,IAAa,CAAC,MAAD,EAAS,UAAT,EAAqB,WAArB,EAAkCG,QAAlC,CAA2C1B,IAA3C,CADf,CADK,EAGJ,EAHI,CAAP;AAID;;AAE4B,aAAlBmB,kBAAkB,GAAG;AAC9B,WAAO,KAAKR,UAAL,CAAgB,+BAAhB,EAAiD,MACtD,KAAKU,aAAL,CAAmB,CAAC;AAAEE,MAAAA;AAAF,KAAD,KAAkBA,QAArC,CADK,EAEJ,EAFI,CAAP;AAGD;;AAE0B,aAAhBI,gBAAgB,GAAG;AAC5B,WAAO,KAAKhB,UAAL,CAAgB,6BAAhB,EAA+C,MACpD,KAAKU,aAAL,CAAmB,CAAC;AAAEO,MAAAA;AAAF,KAAD,KAAgBA,MAAnC,CADK,EAEJ,EAFI,CAAP;AAGD;;AAEmB,SAAbP,aAAa,CAACQ,MAAD,EAAS;AAC3B,UAAMC,UAAU,GAAG,EAAnB;AACA,UAAM;AAAEtG,MAAAA;AAAF,QAAiB,KAAKV,UAA5B;;AACA,SAAK,MAAM,CAACd,IAAD,EAAOyB,QAAP,CAAX,IAA+BpC,MAAM,CAACuF,OAAP,CAAepD,UAAf,CAA/B,EAA2D;AACzD,UAAIqG,MAAM,CAACpG,QAAD,CAAV,EAAsB;AACpBqG,QAAAA,UAAU,CAAChI,IAAX,CAAgBE,IAAhB;AACD;AACF;;AACD,WAAO8H,UAAP;AACD;;AAEgB,SAAVnB,UAAU,CAACoB,UAAD,EAAaC,SAAb,EAAwBC,KAAK,GAAG,EAAhC,EAAoC;AAAA;;AACnD,QAAIC,KAAK,GAAGC,OAAO,CAAC,IAAD,EAAO,OAAP,EAAgB,EAAhB,CAAnB;AAKA,QAAIC,KAAJ;;AACA,SAAK,MAAMC,IAAX,IAAmBN,UAAU,CAACO,KAAX,CAAiB,GAAjB,CAAnB,EAA0C;AACxCF,MAAAA,KAAK,GAAGF,KAAK,CAACG,IAAD,CAAL,GAAcH,KAAK,CAACG,IAAD,CAAL,IAAe;AACnCH,QAAAA,KAAK,EAAE,EAD4B;AAEnC3H,QAAAA,KAAK,EAAEgF;AAF4B,OAArC;AAIA2C,MAAAA,KAAK,GAAGE,KAAK,CAACF,KAAd;AACD;;AACD,QAAI,WAAAE,KAAK,SAAL,mBAAO7H,KAAP,MAAiBgF,SAArB,EAAgC;AAG9B6C,MAAAA,KAAK,CAAC7H,KAAN,GAAc0H,KAAd;AACAG,MAAAA,KAAK,CAAC7H,KAAN,GAAcyH,SAAS,EAAvB;AAEAI,MAAAA,KAAK,CAACF,KAAN,GAAc,EAAd;AACD;;AACD,sBAAOE,KAAP,qBAAO,QAAO7H,KAAd;AACD;;AAEyB,SAAnBV,mBAAmB,GAAG;AAC3B,WAAOsI,OAAO,CAAC,IAAD,EAAO,kBAAP,EAA2B,EAA3B,CAAd;AACD;;AAS8B,SAAxBI,wBAAwB,CAACC,YAAD,EAAe;AAC5C,WAAOA,YAAP;AACD;;AAG8B,SAAxBC,wBAAwB,CAACC,UAAD,EAAa;AAC1C,WAAOA,UAAP;AACD;;AAGDzJ,EAAAA,QAAQ,CAACD,IAAD,EAAOyD,OAAP,EAAgB;AACtBA,IAAAA,OAAO,GAAGA,OAAO,IAAI,EAArB;AACA,UAAMkG,cAAc,GAGlB,CAAClG,OAAO,CAACL,KAAT,IAEA,KAAKnB,WAAL,KAAqBpC,KAAK,CAACoB,SAAN,CAAgBgB,WAFrC,IAIA,CAAC,KAAKlC,WAAL,CAAiB6G,WAAjB,CAA6B5G,IAA7B,CAPH;;AASA,QAAI,CAAC2J,cAAD,IAAmBlG,OAAO,CAACC,cAA/B,EAA+C;AAC7C,YAAMzD,QAAN,CAAeD,IAAf,EAAqByD,OAArB;;AACA,UAAIkG,cAAJ,EAAoB;AAClB,aAAK1H,WAAL;AACD;AACF,KALD,MAKO;AAKL,YAAMhC,QAAN,CAAeD,IAAf,EAAqB,EAAE,GAAGyD,OAAL;AAAcL,QAAAA,KAAK,EAAE;AAArB,OAArB;AACA,WAAKnB,WAAL;AACA,WAAKuB,SAAL,CAAe,IAAf,EAAqBC,OAArB;AACD;;AACD,WAAO,IAAP;AACD;;AAGDmG,EAAAA,mBAAmB,CAAC5J,IAAD,EAAO;AACxB,UAAM;AAAED,MAAAA;AAAF,QAAkB,IAAxB;;AACA,SAAK,MAAMuG,GAAX,IAAkBvG,WAAW,CAAC0I,cAA9B,EAA8C;AAC5C,YAAMoB,IAAI,GAAG7J,IAAI,CAACsG,GAAD,CAAjB;;AACA,UAAIuD,IAAJ,YAAIA,IAAI,CAAEC,WAAV,EAAuB;AACrB9J,QAAAA,IAAI,CAACsG,GAAD,CAAJ,GAAYuD,IAAI,CAACC,WAAL,EAAZ;AACD;AACF;;AACD,QAAI/J,WAAW,CAACgK,QAAZ,EAAJ,EAA4B;AAE1B,WAAK,MAAMzD,GAAX,IAAkBvG,WAAW,CAACyI,iBAA9B,EAAiD;AAC/C,cAAMwB,IAAI,GAAGhK,IAAI,CAACsG,GAAD,CAAjB;;AACA,YAAI0D,IAAI,KAAKzD,SAAb,EAAwB;AACtBvG,UAAAA,IAAI,CAACsG,GAAD,CAAJ,GAAY0D,IAAI,GAAG,CAAH,GAAO,CAAvB;AACD;AACF;AACF;;AAED,SAAK,MAAM1D,GAAX,IAAkBvG,WAAW,CAACoI,kBAA9B,EAAkD;AAChD,aAAOnI,IAAI,CAACsG,GAAD,CAAX;AACD;;AAID,WAAO,MAAMsD,mBAAN,CAA0B5J,IAA1B,CAAP;AACD;;AAGDiK,EAAAA,kBAAkB,CAACjK,IAAD,EAAO;AACvB,UAAM;AAAED,MAAAA;AAAF,QAAkB,IAAxB;AACAC,IAAAA,IAAI,GAAG,MAAMiK,kBAAN,CAAyBjK,IAAzB,CAAP;;AACA,QAAID,WAAW,CAACgK,QAAZ,EAAJ,EAA4B;AAE1B,WAAK,MAAMzD,GAAX,IAAkBvG,WAAW,CAACyI,iBAA9B,EAAiD;AAC/C,cAAMwB,IAAI,GAAGhK,IAAI,CAACsG,GAAD,CAAjB;;AACA,YAAI0D,IAAI,KAAKzD,SAAb,EAAwB;AACtBvG,UAAAA,IAAI,CAACsG,GAAD,CAAJ,GAAY,CAAC,CAAC0D,IAAd;AACD;AACF;AACF;;AAGD,WAAO,KAAKE,UAAL,CAAgBlK,IAAhB,CAAP;AACD;;AAGDkK,EAAAA,UAAU,CAAClK,IAAD,EAAO;AACf,UAAM;AAAED,MAAAA;AAAF,QAAkB,IAAxB;;AACA,SAAK,MAAMuG,GAAX,IAAkBvG,WAAW,CAAC0I,cAA9B,EAA8C;AAC5C,YAAMoB,IAAI,GAAG7J,IAAI,CAACsG,GAAD,CAAjB;;AACA,UAAIuD,IAAI,KAAKtD,SAAb,EAAwB;AACtBvG,QAAAA,IAAI,CAACsG,GAAD,CAAJ,GAAY,sBAASuD,IAAT,IAAiB,IAAIM,IAAJ,CAASN,IAAT,CAAjB,GAAkCA,IAA9C;AACD;AACF;;AAGD,UAAM;AAAEhI,MAAAA;AAAF,QAAa9B,WAAW,CAAC+B,UAA/B;;AACA,QAAID,MAAJ,EAAY;AACV,WAAK,MAAMuI,QAAX,IAAuBvI,MAAvB,EAA+B;AAC7B,cAAMwI,OAAO,GAAGtK,WAAW,CAACoC,GAAZ,CAAgBmI,UAAhB,CAA2BzI,MAAM,CAACuI,QAAD,CAAN,CAAiBC,OAA5C,CAAhB;AACA,cAAME,IAAI,GAAG,gCAAmBvK,IAAnB,EAAyBoK,QAAzB,EAAmC,MAAM,IAAzC,CAAb;;AACA,YAAIG,IAAJ,EAAU;AACR,gBAAMC,mBAAmB,GAAGD,IAAI,IAAI;AAClC,gBAAIA,IAAJ,EAAU;AACR,kBAAI,qBAAQA,IAAR,CAAJ,EAAmB;AACjBA,gBAAAA,IAAI,CAAC9D,OAAL,CAAa+D,mBAAb;AACD,eAFD,MAEO;AACLH,gBAAAA,OAAO,CAACI,gBAAR,CAAyBF,IAAzB;AACD;AACF;AACF,WARD;;AASAC,UAAAA,mBAAmB,CAACD,IAAD,CAAnB;AACD;AACF;AACF;;AACD,WAAOvK,IAAP;AACD;;AAGD0K,EAAAA,WAAW,CAAC1K,IAAD,EAAO;AAChB,UAAM;AAAED,MAAAA;AAAF,QAAkB,IAAxB;;AAEA,SAAK,MAAMuG,GAAX,IAAkBvG,WAAW,CAACoI,kBAA9B,EAAkD;AAGhD,UAAI,EAAE7B,GAAG,IAAItG,IAAT,CAAJ,EAAoB;AAClB,cAAMuB,KAAK,GAAG,KAAK+E,GAAL,CAAd;;AACA,YAAI/E,KAAK,KAAKgF,SAAd,EAAyB;AACvBvG,UAAAA,IAAI,CAACsG,GAAD,CAAJ,GAAY/E,KAAZ;AACD;AACF;AACF;;AAED,SAAK,MAAM+E,GAAX,IAAkBvG,WAAW,CAAC4I,gBAA9B,EAAgD;AAC9C,aAAO3I,IAAI,CAACsG,GAAD,CAAX;AACD;;AACD,WAAOtG,IAAP;AACD;;AAID2K,EAAAA,YAAY,CAACC,UAAD,EAAaC,IAAb,EAAmB;AAC7B,WAAO,wBAAY,KAAK9K,WAAjB,EAA8B6K,UAA9B,EAA0CC,IAA1C,CAAP;AACD;;AAEmB,QAAdC,cAAc,CAACF,UAAD,EAAaC,IAAb,EAAmBlI,GAAnB,EAAwB;AAC1C,WAAO,0BAAc,KAAK5C,WAAnB,EAAgC6K,UAAhC,EAA4CC,IAA5C,EAAkDlI,GAAlD,CAAP;AACD;;AAEiB,SAAXoI,WAAW,CAACH,UAAD,EAAaC,IAAb,EAAmB;AACnC,WAAO,wBAAY,IAAZ,EAAkBD,UAAlB,EAA8BC,IAA9B,CAAP;AACD;;AAEyB,eAAbG,aAAa,CAACJ,UAAD,EAAaC,IAAb,EAAmBlI,GAAnB,EAAwB;AAChD,WAAO,0BAAc,IAAd,EAAoBiI,UAApB,EAAgCC,IAAhC,EAAsClI,GAAtC,CAAP;AACD;;AAEqC,SAA/BsI,+BAA+B,CAACb,QAAD,EAAW;AAG/C,UAAMc,cAAc,GAAG,2BAAcd,QAAd,CAAvB;AACA,QAAIzD,KAAK,GAAG,CAAZ;;AAEA,UAAMwE,SAAS,GAAG,CAAC1I,QAAQ,GAAG,IAAZ,EAAkBrC,QAAQ,GAAG,IAA7B,KAAsC;AACtD,YAAMgL,KAAK,GAAG3I,QAAQ,IAAIrC,QAA1B;AACA,YAAMY,IAAI,GAAGkK,cAAc,CAACvE,KAAD,CAA3B;AACA,YAAM0E,IAAI,GAAG1E,KAAK,GAAG,CAArB;AACA,YAAMyD,QAAQ,GAAGgB,KAAK,GAClB,+BAAkBF,cAAc,CAACI,KAAf,CAAqB,CAArB,EAAwBD,IAAxB,CAAlB,CADkB,GAElB,IAFJ;AAGA,YAAME,cAAc,GAAGH,KAAK,GACxB,+BAAkBF,cAAc,CAACI,KAAf,CAAqBD,IAArB,CAAlB,CADwB,GAExB,IAFJ;AAGA,YAAMG,UAAU,GAAGJ,KAAK,GACpBF,cAAc,CAACI,KAAf,CAAqB,CAArB,EAAwBlL,QAAQ,GAAGiL,IAAH,GAAU1E,KAA1C,EAAiD8E,IAAjD,CAAsD,GAAtD,KACChJ,QAAQ,GAAI,KAAIzB,IAAK,GAAb,GAAkB,EAD3B,CADoB,GAGpB,IAHJ;AAIA,aAAO;AACLyB,QAAAA,QADK;AAELrC,QAAAA,QAFK;AAGLgK,QAAAA,QAHK;AAILmB,QAAAA,cAJK;AAKLvK,QAAAA,IALK;AAMLwK,QAAAA,UANK;AAOL7E,QAAAA;AAPK,OAAP;AASD,KAvBD;;AAyBA,UAAM,CAAC+E,UAAD,EAAa,GAAGC,WAAhB,IAA+BT,cAArC;AACA,UAAMzI,QAAQ,GAAG,KAAKX,UAAL,CAAgBU,UAAhB,CAA2BkJ,UAA3B,CAAjB;;AACA,QAAIjJ,QAAJ,EAAc;AACZ,aAAO0I,SAAS,CAAC1I,QAAD,CAAhB;AACD,KAFD,MAEO;AACL,UAAIrC,QAAQ,GAAG,KAAKG,YAAL,GAAoBmL,UAApB,CAAf;;AACA,UAAItL,QAAJ,EAAc;AACZ,YAAI;AAAEQ,UAAAA;AAAF,YAAwBR,QAA5B;;AACA,aAAK,MAAMwL,KAAX,IAAoBD,WAApB,EAAiC;AAC/BhF,UAAAA,KAAK;AACL,gBAAMlE,QAAQ,GAAG7B,iBAAiB,CAACkB,UAAlB,CAA6BU,UAA7B,CAAwCoJ,KAAxC,CAAjB;;AACA,cAAInJ,QAAJ,EAAc;AACZ,mBAAO0I,SAAS,CAAC1I,QAAD,CAAhB;AACD,WAFD,MAEO,IAAImJ,KAAK,KAAK,GAAd,EAAmB;AACxB,gBAAIxL,QAAQ,CAACyL,UAAT,EAAJ,EAA2B;AAEzB,qBAAOV,SAAS,EAAhB;AACD,aAHD,MAGO;AACL;AACD;AACF,WAPM,MAOA;AAEL/K,YAAAA,QAAQ,GAAGQ,iBAAiB,CAACL,YAAlB,GAAiCqL,KAAjC,CAAX;;AACA,gBAAIxL,QAAJ,EAAc;AACZQ,cAAAA,iBAAiB,GAAGR,QAAQ,CAACQ,iBAA7B;AACD,aAFD,MAEO;AACL,qBAAOuK,SAAS,EAAhB;AACD;AACF;AACF;;AACD,YAAI/K,QAAJ,EAAc;AAEZ,iBAAO+K,SAAS,CAAC,IAAD,EAAO/K,QAAP,CAAhB;AACD;AACF;AACF;;AACD,WAAO+K,SAAS,EAAhB;AACD;;AAGkB,SAAZW,YAAY,CAACC,YAAD,EAAepJ,GAAf,EAAoB;AAErC,WAAO,MAAMmJ,YAAN,CAAmBC,YAAnB,EAAiCpJ,GAAjC,EAAsCqJ,KAAtC,CAA4CD,YAA5C,CAAP;AACD;;AAGsB,SAAhBE,gBAAgB,CAACjJ,KAAD,EAAQkJ,QAAR,EAAkB;AACvC,QAAI,sBAASA,QAAT,CAAJ,EAAwB;AACtB,UAAIlJ,KAAK,CAACmJ,UAAN,GAAmB5E,QAAnB,CAA4B2E,QAA5B,CAAJ,EAA2C;AACzC,eAAOlJ,KAAK,CAACoJ,UAAN,CAAiBF,QAAjB,CAAP;AACD;;AAED,cAAQA,QAAQ,CAAC,CAAD,CAAhB;AACA,aAAK,GAAL;AAIE,iBAAOlJ,KAAK,CAACoJ,UAAN,CAAiBF,QAAjB,CAAP;;AACF,aAAK,GAAL;AACE,iBAAOlJ,KAAK,CAACqJ,WAAN,CAAkBH,QAAQ,CAACZ,KAAT,CAAe,CAAf,CAAlB,CAAP;;AACF,aAAK,GAAL;AACE,iBAAOtI,KAAK,CAACsJ,MAAN,CAAaJ,QAAQ,CAACZ,KAAT,CAAe,CAAf,CAAb,CAAP;AATF;AAWD;;AACD,UAAMW,gBAAN,CAAuBjJ,KAAvB,EAA8BkJ,QAA9B;AACD;;AAGyB,SAAnBK,mBAAmB,CAAClI,GAAD,EAAM5D,KAAN,EAAa;AACrC,WAAO,IAAI+L,qBAAJ,CACL/L,KAAK,KACH4D,GAAG,CAACoI,IAAJ,GACK,IAAG,KAAKzL,IAAK,mBAAkBqD,GAAG,CAACoI,IAAK,YAD7C,GAEK,IAAG,KAAKzL,IAAK,mBAHf,CADA,CAAP;AAOD;;AAGqB,SAAf0L,eAAe,GAAG;AAIvB,WAAO,KAAKvK,GAAL,CAAS+B,SAAhB;AACD;;AAG2B,SAArByI,qBAAqB,CAAC;AAAE3F,IAAAA,IAAF;AAAQ4F,IAAAA,OAAR;AAAiBC,IAAAA,MAAjB;AAAyBpJ,IAAAA,OAAzB;AAAkCzD,IAAAA;AAAlC,GAAD,EAA2C;AACrE,YAAQgH,IAAR;AACA,WAAK,iBAAL;AACE,eAAO,KAAK7E,GAAL,CAASwK,qBAAT,CAA+B;AACpC3F,UAAAA,IADoC;AAEpC4F,UAAAA,OAAO,EACLA,OAAO,IAAK,6BAA4B,KAAK5L,IAAK,qBAHhB;AAIpC6L,UAAAA,MAJoC;AAKpCpJ,UAAAA,OALoC;AAMpCzD,UAAAA;AANoC,SAA/B,CAAP;;AAQF,WAAK,oBAAL;AACA,WAAK,mBAAL;AACE,eAAO,IAAIU,qBAAJ,CAAkB;AAAEsG,UAAAA,IAAF;AAAQ4F,UAAAA,OAAR;AAAiBC,UAAAA;AAAjB,SAAlB,CAAP;;AACF,WAAK,cAAL;AACE,eAAO,IAAIC,kBAAJ,CAAe;AAAE9F,UAAAA,IAAF;AAAQ4F,UAAAA,OAAR;AAAiBC,UAAAA;AAAjB,SAAf,CAAP;;AACF;AACE,eAAO,IAAI7H,qBAAJ,CAAkB;AAAEgC,UAAAA,IAAF;AAAQ4F,UAAAA,OAAR;AAAiBC,UAAAA;AAAjB,SAAlB,CAAP;AAhBF;AAkBD;;AAcoB,aAAV/K,UAAU,GAAG;AAEtB,WAAOqH,OAAO,CAAC,IAAD,EAAO,YAAP,EAAqB,MAAM;AACvC,YAAMrH,UAAU,GAAG,EAAnB;;AAEA,YAAMiL,aAAa,GAAG,CAAC/L,IAAD,EAAOyB,QAAP,KAAoB;AACxCpC,QAAAA,MAAM,CAACgB,cAAP,CAAsBS,UAAtB,EAAkCd,IAAlC,EAAwC,EACtC,GAAGyB,QADmC;AAEtCf,UAAAA,UAAU,EAAE;AAF0B,SAAxC;AAID,OALD;;AAOA,YAAMsL,aAAa,GAAGhM,IAAI,IAAI;AAC5B,YAAImL,UAAU,GAAG,IAAjB;AAOA,cAAM7L,MAAM,GAAG,EAAf;;AACA,eAAO6L,UAAU,KAAKrM,mBAAUD,KAAhC,EAAuC;AAErC,cAAImB,IAAI,IAAImL,UAAZ,EAAwB;AAKtB,kBAAMc,IAAI,GAAG5M,MAAM,CAAC6M,wBAAP,CAAgCf,UAAhC,EAA4CnL,IAA5C,CAAb;;AACA,gBAAIiM,IAAJ,EAAU;AAAA;;AACR,oBAAM1L,KAAK,GAAG,cAAA0L,IAAI,CAAC3L,GAAL,+BAAU6L,IAAV,CAAe,IAAf,MAAwBF,IAAI,CAAC1L,KAA3C;;AACA,kBAAIA,KAAJ,EAAW;AACTjB,gBAAAA,MAAM,CAACQ,IAAP,CAAYS,KAAZ;AACD;AACF;AACF;;AACD4K,UAAAA,UAAU,GAAG9L,MAAM,CAAC+M,cAAP,CAAsBjB,UAAtB,CAAb;AACD;;AAGDY,QAAAA,aAAa,CAAC/L,IAAD,EAAO;AAClBS,UAAAA,YAAY,EAAE,IADI;AAElBF,UAAAA,KAAK,EAAE;AAFW,SAAP,CAAb;;AAIA,YAAI;AACF,gBAAM8L,MAAM,GAAGC,qBAAYtM,IAAZ,EAAkBmM,IAAlB,CAAuB,IAAvB,EAA6B7M,MAA7B,CAAf;;AAEAyM,UAAAA,aAAa,CAAC/L,IAAD,EAAO;AAClBS,YAAAA,YAAY,EAAE,KADI;AAElBF,YAAAA,KAAK,EAAE8L;AAFW,WAAP,CAAb;AAIA,iBAAOA,MAAP;AACD,SARD,CAQE,OAAO5M,KAAP,EAAc;AACd,gBAAM,IAAIiG,kBAAJ,CAAe,IAAf,EAAqBjG,KAAK,CAACmM,OAA3B,CAAN;AACD;AACF,OA3CD;;AAiDA,WAAK,MAAM5L,IAAX,IAAmBsM,oBAAnB,EAAgC;AAC9BP,QAAAA,aAAa,CAAC/L,IAAD,EAAO;AAClBS,UAAAA,YAAY,EAAE,IADI;AAElBH,UAAAA,GAAG,EAAE,MAAM0L,aAAa,CAAChM,IAAD;AAFN,SAAP,CAAb;AAID;;AACD,aAAOc,UAAP;AACD,KAlEa,CAAd;AAmED;;AAIDyL,EAAAA,KAAK,CAACC,KAAD,EAAQ,GAAGpJ,IAAX,EAAiB;AACpB,WAAO,KAAKrE,WAAL,CAAiB0N,IAAjB,CAAsBD,KAAtB,EAA6B,IAA7B,EAAmC,GAAGpJ,IAAtC,CAAP;AACD;;AAEgB,SAAVsJ,UAAU,CAACtJ,IAAD,EAAO;AACtB,WAAO,KAAKuJ,eAAL,CAAqB,aAArB,EAAoCvJ,IAApC,CAAP;AACD;;AAEe,SAATwJ,SAAS,CAACxJ,IAAD,EAAO;AACrB,WAAO,KAAKuJ,eAAL,CAAqB,YAArB,EAAmCvJ,IAAnC,CAAP;AACD;;AAEkB,SAAZyJ,YAAY,CAACzJ,IAAD,EAAO;AACxB,WAAO,KAAKuJ,eAAL,CAAqB,eAArB,EAAsCvJ,IAAtC,CAAP;AACD;;AAEiB,SAAX0J,WAAW,CAAC1J,IAAD,EAAO;AACvB,WAAO,KAAKuJ,eAAL,CAAqB,cAArB,EAAqCvJ,IAArC,CAAP;AACD;;AAEkB,SAAZ2J,YAAY,CAAC3J,IAAD,EAAO;AACxB,WAAO,KAAKuJ,eAAL,CAAqB,eAArB,EAAsCvJ,IAAtC,CAAP;AACD;;AAEiB,SAAX4J,WAAW,CAAC5J,IAAD,EAAO;AACvB,WAAO,KAAKuJ,eAAL,CAAqB,cAArB,EAAqCvJ,IAArC,CAAP;AACD;;AAEkB,SAAZ6J,YAAY,CAAC7J,IAAD,EAAO;AACxB,WAAO,KAAKuJ,eAAL,CAAqB,eAArB,EAAsCvJ,IAAtC,CAAP;AACD;;AAEiB,SAAX8J,WAAW,CAAC9J,IAAD,EAAO;AACvB,WAAO,KAAKuJ,eAAL,CAAqB,cAArB,EAAqCvJ,IAArC,CAAP;AACD;;AAE2B,eAAfuJ,eAAe,CAACH,KAAD,EAAQW,YAAR,EAAsB;AAChD,UAAMC,SAAS,GAAG,KAAKA,SAAL,CAAeZ,KAAf,CAAlB;;AACA,QAAIY,SAAS,CAACtI,MAAV,GAAmB,CAAvB,EAA0B;AAGxB,UAAI;AAAE/C,QAAAA;AAAF,UAAaoL,YAAjB;AAIA,YAAM/J,IAAI,GAAG/D,MAAM,CAACiE,MAAP,CAAc6J,YAAd,EAA4B;AACvCnH,QAAAA,IAAI,EAAE;AACJzF,UAAAA,KAAK,EAAEiM;AADH,SADiC;AAIvCzK,QAAAA,MAAM,EAAE;AACNzB,UAAAA,GAAG,GAAG;AACJ,mBAAOyB,MAAP;AACD;;AAHK;AAJ+B,OAA5B,CAAb;;AAUA,WAAK,MAAMsL,QAAX,IAAuBD,SAAvB,EAAkC;AAChC,cAAME,GAAG,GAAG,MAAMD,QAAQ,CAAClB,IAAT,CAAc,IAAd,EAAoB/I,IAApB,CAAlB;;AACA,YAAIkK,GAAG,KAAK/H,SAAZ,EAAuB;AACrBxD,UAAAA,MAAM,GAAGuL,GAAT;AACD;AACF;;AAID,UAAIvL,MAAM,KAAKoL,YAAY,CAACpL,MAA5B,EAAoC;AAClC,eAAOA,MAAP;AACD;AACF;AACF;;AAIwB,SAAlBf,kBAAkB,CAACH,MAAD,EAAS;AAChC,UAAM0M,cAAc,GAAGlO,MAAM,CAACmO,IAAP,CAAY3M,MAAZ,CAAvB;AAEA,SAAK4M,EAAL,CAAQ,CACN,eADM,EAEN,eAFM,EAGN,eAHM,CAAR,EAIG,OAAO;AAAEzH,MAAAA,IAAF;AAAQzD,MAAAA,WAAR;AAAqBmL,MAAAA,UAArB;AAAiCC,MAAAA;AAAjC,KAAP,KAA0D;AAC3D,YAAMC,UAAU,GAAG5H,IAAI,KAAK,eAAT,GACf,EADe,GAEf0H,UAFJ;AAKA,YAAMG,SAAS,GAAGD,UAAU,CAAC9I,MAAX,GAAoB,CAApB,GACdyI,cAAc,CAAC1F,MAAf,CACAiG,IAAI,IAAIC,uBAAuB,CAACH,UAAU,CAAC,CAAD,CAAX,EAAgBE,IAAhB,CAAvB,KAAiDvI,SADzD,CADc,GAIdgI,cAJJ;;AAQA,UAAIM,SAAS,CAAC/I,MAAV,KAAqB,CAAzB,EAA4B;AAC1B;AACD;;AAID,YAAMkJ,WAAW,GAAGhI,IAAI,KAAK,eAAT,GAChB,EADgB,GAEhB,MAAMiI,kBAAkB,CAACN,WAAW,EAAZ,EAAgBE,SAAhB,CAF5B;AAGA,YAAMK,sBAAsB,GAAGC,wBAAwB,CACrDH,WADqD,EAErDH,SAFqD,CAAvD;AAIA,YAAMO,qBAAqB,GAAGD,wBAAwB,CACpDP,UADoD,EAEpDC,SAFoD,CAAtD;AAKA,YAAMQ,aAAa,GAAG,EAAtB;AACA,YAAMC,aAAa,GAAG,EAAtB;;AAEA,UAAI/L,WAAW,CAACgM,QAAhB,EAA0B;AAGxBhM,QAAAA,WAAW,CAACiM,eAAZ,CAA4B,CAA5B;AACAjM,QAAAA,WAAW,CAACkL,EAAZ,CAAe,UAAf,EAA2B,MAAMhO,KAAN,IAAe;AACxC,cAAI4O,aAAa,CAACvJ,MAAd,GAAuB,CAA3B,EAA8B;AAC5B2J,YAAAA,OAAO,CAACC,IAAR,CACG,aAAYjP,KAAM,qCACjB4O,aAAa,CAACM,GAAd,CAAkBC,IAAI,IAAK,IAAGA,IAAI,CAAC5O,IAAK,GAAxC,CACD,EAHH;AAKA,kBAAMoE,OAAO,CAACyK,GAAR,CACJR,aAAa,CAACM,GAAd,CACEC,IAAI,IAAIA,IAAI,CAACvF,OAAL,CAAayF,UAAb,CAAwBF,IAAxB,CADV,CADI,CAAN;AAKD;;AACD,cAAIN,aAAa,CAACxJ,MAAd,GAAuB,CAA3B,EAA8B;AAG5B2J,YAAAA,OAAO,CAACC,IAAR,CACG,mDACCJ,aAAa,CAACK,GAAd,CAAkBC,IAAI,IAAK,IAAGA,IAAI,CAAC5O,IAAK,GAAxC,CACD,EAHH;AAKD;AACF,SAtBD;AAuBD;;AAED,WAAK,MAAMoJ,QAAX,IAAuByE,SAAvB,EAAkC;AAChC,cAAMxE,OAAO,GAAG,KAAKlI,GAAL,CAASmI,UAAT,CAAoBzI,MAAM,CAACuI,QAAD,CAAN,CAAiBC,OAArC,CAAhB;AACA,cAAM0F,WAAW,GAAGb,sBAAsB,CAAC9E,QAAD,CAAtB,IAAoC,EAAxD;AACA,cAAM4F,UAAU,GAAGZ,qBAAqB,CAAChF,QAAD,CAArB,IAAmC,EAAtD;AACA,cAAM6F,WAAW,GAAGC,aAAa,CAACH,WAAD,CAAjC;AACA,cAAMI,UAAU,GAAGD,aAAa,CAACF,UAAD,CAAhC;AACA,cAAMI,YAAY,GAAGL,WAAW,CAAClH,MAAZ,CAAmB+G,IAAI,IAAI,CAACO,UAAU,CAACP,IAAI,CAACtJ,GAAN,CAAtC,CAArB;AACA,cAAM+J,UAAU,GAAGL,UAAU,CAACnH,MAAX,CAAkB+G,IAAI,IAAI,CAACK,WAAW,CAACL,IAAI,CAACtJ,GAAN,CAAtC,CAAnB;AAKA,cAAMgJ,aAAa,GAAGU,UAAU,CAACnH,MAAX,CACpB+G,IAAI,IAAIA,IAAI,CAACrF,IAAL,IAAa0F,WAAW,CAACL,IAAI,CAACtJ,GAAN,CADZ,CAAtB;AAGA+I,QAAAA,aAAa,CAACvO,IAAd,CACE,IAAG,MAAM,KAAKqB,GAAL,CAASmO,4BAAT,CACPjG,OADO,EAEPgG,UAFO,EAGPD,YAHO,EAIP7M,WAJO,CAAT,CADF;AAQA+L,QAAAA,aAAa,CAACxO,IAAd,CACE,IAAG,MAAM,KAAKqB,GAAL,CAASoO,oBAAT,CACPlG,OADO,EAEPiF,aAFO,EAGP/L,WAHO,CAAT,CADF;AAOD;AACF,KAnGD;AAoGD;;AA/9BwC;;;AAA9B1D,K,CA4tBJ2Q,Y,GAAeA,mB;AA5tBX3Q,K,CA+tBJ4Q,qB,GAAwB,K;AA/tBpB5Q,K,CAkuBJ6Q,wB,GAA2B,I;AAluBvB7Q,K,CAquBJ8Q,e,GAAkB,I;;AA6P3BC,kBAAaC,KAAb,CAAmBhR,KAAnB;;AACAiR,gBAAWD,KAAX,CAAiBhR,KAAjB;;AAEA2Q,oBAAaK,KAAb,CAAmBhR,KAAnB;;AAEA,MAAMkR,OAAO,GAAG,IAAIC,OAAJ,EAAhB;;AAEA,SAAS7H,OAAT,CAAiBgD,UAAjB,EAA6B7F,GAA7B,EAAkC/E,KAAlC,EAAyC;AACvC,MAAI0P,IAAI,GAAGF,OAAO,CAACzP,GAAR,CAAY6K,UAAZ,CAAX;;AACA,MAAI,CAAC8E,IAAL,EAAW;AACTF,IAAAA,OAAO,CAACG,GAAR,CAAY/E,UAAZ,EAAwB8E,IAAI,GAAG,EAA/B;AACD;;AACD,MAAI,EAAE3K,GAAG,IAAI2K,IAAT,CAAJ,EAAoB;AAClBA,IAAAA,IAAI,CAAC3K,GAAD,CAAJ,GAAY,wBAAW/E,KAAX,IAAoBA,KAAK,EAAzB,GAA8BA,KAA1C;AACD;;AACD,SAAO0P,IAAI,CAAC3K,GAAD,CAAX;AACD;;AAED,SAAS2I,kBAAT,CAA4BjM,KAA5B,EAAmC6L,SAAnC,EAA8C;AAC5C,SAAOA,SAAS,CAAC5H,MAAV,CACL,CAACjE,KAAD,EAAQoH,QAAR,KAAqBpH,KAAK,CAACmO,YAAN,CAAmB/G,QAAnB,CADhB,EAELpH,KAFK,CAAP;AAID;;AAED,SAAS+L,uBAAT,CAAiCqC,IAAjC,EAAuCtC,IAAvC,EAA6C;AAC3C,SAAO,gCAAmBsC,IAAnB,EAAyBtC,IAAzB,EAA+B,MAAMvI,SAArC,CAAP;AACD;;AAED,SAAS4I,wBAAT,CAAkCkC,KAAlC,EAAyCxC,SAAzC,EAAoD;AAClD,SAAOA,SAAS,CAAC5H,MAAV,CACL,CAACqK,QAAD,EAAWlH,QAAX,KAAwB;AACtBkH,IAAAA,QAAQ,CAAClH,QAAD,CAAR,GAAqB,qBAAQiH,KAAR,EAAepK,MAAf,CACnB,CAACsK,KAAD,EAAQH,IAAR,KAAiB;AACf,YAAM7G,IAAI,GAAG,qBAAQwE,uBAAuB,CAACqC,IAAD,EAAOhH,QAAP,CAA/B,CAAb;AAGAmH,MAAAA,KAAK,CAACzQ,IAAN,CAAW,GAAG,qBAAQyJ,IAAR,EAAc1B,MAAd,CAAqB+G,IAAI,IAAI,CAAC,CAACA,IAA/B,CAAd;AACA,aAAO2B,KAAP;AACD,KAPkB,EAQnB,EARmB,CAArB;AAUA,WAAOD,QAAP;AACD,GAbI,EAcL,EAdK,CAAP;AAgBD;;AAED,SAASpB,aAAT,CAAuBqB,KAAvB,EAA8B;AAC5B,SAAOA,KAAK,CAACtK,MAAN,CACL,CAAC0I,GAAD,EAAMC,IAAN,KAAe;AACbD,IAAAA,GAAG,CAACC,IAAI,CAACtJ,GAAN,CAAH,GAAgBsJ,IAAhB;AACA,WAAOD,GAAP;AACD,GAJI,EAKL,EALK,CAAP;AAOD","sourcesContent":["import objection from 'objection'\nimport { QueryBuilder } from '@/query'\nimport { EventEmitter, KnexHelper } from '@/lib'\nimport { convertSchema, addRelationSchemas, convertRelations } from '@/schema'\nimport { populateGraph, filterGraph } from '@/graph'\nimport { formatJson } from '@/utils'\nimport {\n  ResponseError,\n  GraphError, ModelError,\n  NotFoundError,\n  RelationError,\n  WrappedError\n} from '@/errors'\nimport {\n  isString, isObject, isArray, isFunction, isPromise, asArray, merge, flatten,\n  parseDataPath, normalizeDataPath, getValueAtDataPath\n} from '@ditojs/utils'\nimport RelationAccessor from './RelationAccessor'\nimport definitions from './definitions'\n\nexport class Model extends objection.Model {\n  // Define a default constructor to allow new Model(json) as a short-cut to\n  // `Model.fromJson(json, { skipValidation: true })`\n  constructor(json) {\n    super()\n    if (json) {\n      this.$setJson(json)\n    }\n  }\n\n  static setup(knex) {\n    this.knex(knex)\n    try {\n      for (const relation of Object.values(this.getRelations())) {\n        this.setupRelation(relation)\n      }\n    } catch (error) {\n      throw error instanceof RelationError ? error : new RelationError(error)\n    }\n    this.referenceValidator = null\n  }\n\n  static setupRelation(relation) {\n    // Add this relation to the related model's relatedRelations, so it can\n    // register all required foreign keys in its properties.\n    relation.relatedModelClass.getRelatedRelations().push(relation)\n    // TODO: Check `through` settings to make sure they're correct?\n\n    // Expose RelationAccessor instances for each relation under short-cut $name\n    // for access to relations and implicit calls to $relatedQuery(name).\n    const accessor = `$${relation.name}`\n    if (accessor in this.prototype) {\n      throw new RelationError(\n        `Model '${this.name}' already defines a property with name ` +\n        `'${accessor}' that clashes with the relation accessor.`)\n    }\n\n    // Define an accessor on the class as well as on the prototype that when\n    // first called creates a RelationAccessor instance and then overrides the\n    // accessor with one that then just returns the same value afterwards.\n    const defineAccessor = (target, isClass) => {\n      Object.defineProperty(target, accessor, {\n        get() {\n          const value = new RelationAccessor(\n            relation,\n            isClass ? this : null, // modelClass\n            isClass ? null : this // model\n          )\n          // Override accessor with value on first call for caching.\n          Object.defineProperty(this, accessor, {\n            value,\n            configurable: true,\n            enumerable: false\n          })\n          return value\n        },\n        configurable: true,\n        enumerable: false\n      })\n    }\n\n    defineAccessor(this, true)\n    defineAccessor(this.prototype, false)\n  }\n\n  // @overridable\n  static initialize() {\n    const { hooks, assets } = this.definition\n    this._setupEmitter(hooks)\n    if (assets) {\n      this._setupAssetsEvents(assets)\n    }\n  }\n\n  // @overridable\n  $initialize() {\n  }\n\n  get $app() {\n    return this.constructor.app\n  }\n\n  $is(model) {\n    return model?.constructor === this.constructor && model?.id === this.id\n  }\n\n  $has(...properties) {\n    for (const property of properties) {\n      if (!(property in this)) return false\n    }\n    return true\n  }\n\n  $update(properties, trx) {\n    return this.$query(trx)\n      .update(properties)\n      .runAfter((result, query) =>\n        // Only perform `$set()` and return `this` if the query wasn't modified\n        // in a way that would remove the `update()` command, e.g. toFindQuery()\n        query.has('update') ? this.$set(result) : result\n      )\n  }\n\n  $patch(properties, trx) {\n    return this.$query(trx)\n      .patch(properties)\n      .runAfter((result, query) =>\n        // Only perform `$set()` and return `this` if the query wasn't modified\n        // in a way that would remove the `patch()` command, e.g. toFindQuery()\n        query.has('patch') ? this.$set(result) : result\n      )\n  }\n\n  // @override\n  $transaction(trx, handler) {\n    return this.constructor.transaction(trx, handler)\n  }\n\n  // @override\n  static transaction(trx, handler) {\n    // Support both `transaction(trx, handler)` & `transaction(handler)`\n    if (!handler) {\n      handler = trx\n      trx = null\n    }\n    if (handler) {\n      // Use existing transaction, or create new one, to execute handler with:\n      return trx\n        ? handler(trx)\n        : this.knex().transaction(handler)\n    } else {\n      // No arguments, simply delegate to objection's transaction()\n      return super.transaction()\n    }\n  }\n\n  // @override\n  $validate(json, options = {}) {\n    if (options.skipValidation) {\n      return json\n    }\n    if (!options.graph && !options.async) {\n      // Fall back to Objection's $validate() if we don't need any of our\n      // extensions (async and graph for now):\n      return super.$validate(json, options)\n    }\n    json = json || this\n    const inputJson = json\n    const shallow = json.$isObjectionModel && !options.graph\n    if (shallow) {\n      // Strip away relations and other internal stuff.\n      json = json.clone({ shallow: true })\n      // We can mutate `json` now that we took a copy of it.\n      options = { ...options, mutable: true }\n    }\n\n    const validator = this.constructor.getValidator()\n    const args = {\n      options,\n      model: this,\n      json,\n      ctx: Object.create(null)\n    }\n\n    validator.beforeValidate(args)\n    const result = validator.validate(args)\n    const handleResult = result => {\n      validator.afterValidate(args)\n      // If `json` was shallow-cloned, copy over the possible default values.\n      return shallow ? inputJson.$set(result) : result\n    }\n    // Handle both async and sync validation here:\n    return isPromise(result)\n      ? result.then(handleResult)\n      : handleResult(result)\n  }\n\n  async $validateGraph(options = {}) {\n    await this.$validate(null, {\n      ...options,\n      graph: true,\n      // Always use `async: true` option here for simplicity:\n      async: true\n    })\n    return this\n  }\n\n  // @override\n  static fromJson(json, options = {}) {\n    if (options.async && !options.skipValidation) {\n      // Handle async validation, as supported by Dito:\n      const model = new this()\n      return model.$validate(json, options).then(\n        json => model.$setJson(json, {\n          ...options,\n          skipValidation: true\n        })\n      )\n    }\n    // Fall back to Objection's fromJson() if we don't need async handling:\n    return super.fromJson(json, options)\n  }\n\n  // @override\n  static query(trx) {\n    return super.query(trx).onError(err => {\n      // TODO: Shouldn't this wrapping happen on the Controller level?\n      err = err instanceof ResponseError ? err\n        : err instanceof objection.DBError ? this.app.createDatabaseError(err)\n        : new WrappedError(err)\n      return Promise.reject(err)\n    })\n  }\n\n  static async count(...args) {\n    const { count } = await this.query().count(...args).first() || {}\n    return +count || 0\n  }\n\n  // @override\n  static get tableName() {\n    // If the class name ends in 'Model', remove that from the table name.\n    return this.name.match(/^(.*?)(?:Model|)$/)[1]\n  }\n\n  // @override\n  static get idColumn() {\n    // Try extracting the id column name from the raw properties definitions,\n    // not the resolved `definition.properties` which aren't ready at this point\n    // with fall-back onto default Objection.js behavior.\n    const { properties } = this\n    const ids = []\n    for (const [name, property] of Object.entries(properties || {})) {\n      if (property?.primary) {\n        ids.push(name)\n      }\n    }\n    const { length } = ids\n    return length > 1 ? ids : length > 0 ? ids[0] : super.idColumn\n  }\n\n  static getReference(modelOrId, includeProperties) {\n    // Creates a reference model that takes over the id / #ref properties from\n    // the passed model or id value/array, omitting any other properties in it,\n    // except for anything mentioned in the optional `includeProperties` arg.\n    const ref = new this()\n    const idProperties = this.getIdPropertyArray()\n    if (isObject(modelOrId)) {\n      const addProperty = key => {\n        const value = modelOrId[key]\n        if (value !== undefined) {\n          ref[key] = value\n        }\n      }\n      // Also support Objection's #ref type references next to id properties.\n      addProperty(this.uidRefProp)\n      idProperties.forEach(addProperty)\n      includeProperties?.forEach(addProperty)\n    } else {\n      // An id value/array: Map it to the properties in `getIdPropertyArray()`:\n      const ids = asArray(modelOrId)\n      if (ids.length !== idProperties.length) {\n        throw new ModelError(\n          this,\n          `Invalid amount of id values provided for reference: Unable to map ${\n            formatJson(modelOrId, false)\n          } to ${\n            formatJson(idProperties, false)\n          }.`\n        )\n      }\n      idProperties.forEach((key, index) => {\n        ref[key] = ids[index]\n      })\n    }\n    return ref\n  }\n\n  static isReference(obj) {\n    let validator = this.referenceValidator\n    if (!validator) {\n      // For `data` to be considered a reference, it needs to hold only one\n      // value that is either the target's id, or an Objection.js #ref value:\n      validator = this.referenceValidator = this.app.compileValidator(\n        {\n          oneOf: [\n            {\n              type: 'object',\n              // Support composite keys and add a property for each key:\n              properties: this.getIdPropertyArray().reduce(\n                (idProperties, idProperty) => {\n                  idProperties[idProperty] = {\n                    type: this.definition.properties[idProperty].type\n                  }\n                  return idProperties\n                },\n                {}\n              ),\n              additionalProperties: false\n            },\n            {\n              type: 'object',\n              properties: {\n                [this.uidRefProp]: {\n                  type: 'string'\n                }\n              },\n              additionalProperties: false\n            }\n          ]\n        },\n        // Receive `false` instead of thrown exceptions when validation fails:\n        { throw: false }\n      )\n    }\n    return validator(obj)\n  }\n\n  static getScope(name) {\n    return this.definition.scopes[name]\n  }\n\n  static hasScope(name) {\n    return !!this.getScope(name)\n  }\n\n  static getModifiers() {\n    return this.definition.modifiers\n  }\n\n  static get relationMappings() {\n    return this._getCached('relationMappings', () => (\n      convertRelations(this, this.definition.relations, this.app.models)\n    ), {})\n  }\n\n  static get jsonSchema() {\n    return this._getCached('jsonSchema', () => {\n      const schema = convertSchema(this.definition.properties)\n      addRelationSchemas(this, schema.properties)\n      // Merge in root-level schema additions\n      merge(schema, this.definition.schema)\n      return {\n        $id: this.name,\n        $schema: 'http://json-schema.org/draft-07/schema',\n        ...schema\n      }\n    }, {})\n  }\n\n  static get virtualAttributes() {\n    // Leverage Objection's own mechanism called `virtualAttributes` to handle\n    // `computedAttributes` when setting JSON data.\n    return this.computedAttributes\n  }\n\n  static get jsonAttributes() {\n    return this._getCached('jsonSchema:jsonAttributes', () => (\n      this.getAttributes(({ type, specificType, computed }) =>\n        !computed && !specificType && (type === 'object' || type === 'array'))\n    ), [])\n  }\n\n  static get booleanAttributes() {\n    return this._getCached('jsonSchema:booleanAttributes', () => (\n      this.getAttributes(({ type, computed }) =>\n        !computed && type === 'boolean')\n    ), [])\n  }\n\n  static get dateAttributes() {\n    return this._getCached('jsonSchema:dateAttributes', () => (\n      this.getAttributes(({ type, computed }) =>\n        !computed && ['date', 'datetime', 'timestamp'].includes(type))\n    ), [])\n  }\n\n  static get computedAttributes() {\n    return this._getCached('jsonSchema:computedAttributes', () => (\n      this.getAttributes(({ computed }) => computed)\n    ), [])\n  }\n\n  static get hiddenAttributes() {\n    return this._getCached('jsonSchema:hiddenAttributes', () => (\n      this.getAttributes(({ hidden }) => hidden)\n    ), [])\n  }\n\n  static getAttributes(filter) {\n    const attributes = []\n    const { properties } = this.definition\n    for (const [name, property] of Object.entries(properties)) {\n      if (filter(property)) {\n        attributes.push(name)\n      }\n    }\n    return attributes\n  }\n\n  static _getCached(identifier, calculate, empty = {}) {\n    let cache = getMeta(this, 'cache', {})\n    // Use a simple dependency tracking mechanism with cache identifiers that\n    // can be children of other cached values, e.g.:\n    // 'jsonSchema:jsonAttributes' as a child of 'jsonSchema', so that whenever\n    // 'jsonSchema' changes, all cached child values  are invalidated.\n    let entry\n    for (const part of identifier.split(':')) {\n      entry = cache[part] = cache[part] || {\n        cache: {},\n        value: undefined\n      }\n      cache = entry.cache\n    }\n    if (entry?.value === undefined) {\n      // Temporarily set cache to an empty object to prevent endless\n      // recursion with interdependent jsonSchema related calls...\n      entry.value = empty\n      entry.value = calculate()\n      // Clear child dependencies once parent value has changed:\n      entry.cache = {}\n    }\n    return entry?.value\n  }\n\n  static getRelatedRelations() {\n    return getMeta(this, 'relatedRelations', [])\n  }\n\n  // Override propertyNameToColumnName() / columnNameToPropertyName() to not\n  // rely on $formatDatabaseJson() /  $parseDatabaseJson() do detect naming\n  // conventions but assume simply that they're always the same.\n  // This is fine since we can now change naming at Knex level.\n  // See knexSnakeCaseMappers()\n\n  // @override\n  static propertyNameToColumnName(propertyName) {\n    return propertyName\n  }\n\n  // @override\n  static columnNameToPropertyName(columnName) {\n    return columnName\n  }\n\n  // @override\n  $setJson(json, options) {\n    options = options || {}\n    const callInitialize = (\n      // Only call initialize when:\n      // 1. we're not partially patching:\n      !options.patch &&\n      // 2. $initialize() is actually doing something:\n      this.$initialize !== Model.prototype.$initialize &&\n      // 3. the data is not just a reference:\n      !this.constructor.isReference(json)\n    )\n    if (!callInitialize || options.skipValidation) {\n      super.$setJson(json, options)\n      if (callInitialize) {\n        this.$initialize()\n      }\n    } else {\n      // If validation isn't skipped or the model provides its own $initialize()\n      // method, call $setJson() with patch validation first to not complain\n      // about missing fields, then perform a full validation after calling\n      // $initialize(), to give the model a chance to configure itself.\n      super.$setJson(json, { ...options, patch: true })\n      this.$initialize()\n      this.$validate(this, options)\n    }\n    return this\n  }\n\n  // @override\n  $formatDatabaseJson(json) {\n    const { constructor } = this\n    for (const key of constructor.dateAttributes) {\n      const date = json[key]\n      if (date?.toISOString) {\n        json[key] = date.toISOString()\n      }\n    }\n    if (constructor.isSQLite()) {\n      // SQLite does not support boolean natively and needs conversion...\n      for (const key of constructor.booleanAttributes) {\n        const bool = json[key]\n        if (bool !== undefined) {\n          json[key] = bool ? 1 : 0\n        }\n      }\n    }\n    // Remove the computed properties so they don't attempt to get set.\n    for (const key of constructor.computedAttributes) {\n      delete json[key]\n    }\n    // NOTE: No need to normalize the identifiers in the JSON in case of\n    // normalizeDbNames, as this already happens through\n    // knex.config.wrapIdentifier(), see Application.js\n    return super.$formatDatabaseJson(json)\n  }\n\n  // @override\n  $parseDatabaseJson(json) {\n    const { constructor } = this\n    json = super.$parseDatabaseJson(json)\n    if (constructor.isSQLite()) {\n      // SQLite does not support boolean natively and needs conversion...\n      for (const key of constructor.booleanAttributes) {\n        const bool = json[key]\n        if (bool !== undefined) {\n          json[key] = !!bool\n        }\n      }\n    }\n    // Also run through normal $parseJson(), for handling of `Date` and\n    // `AssetFile`.\n    return this.$parseJson(json)\n  }\n\n  // @override\n  $parseJson(json) {\n    const { constructor } = this\n    for (const key of constructor.dateAttributes) {\n      const date = json[key]\n      if (date !== undefined) {\n        json[key] = isString(date) ? new Date(date) : date\n      }\n    }\n    // Convert plain asset files objects to AssetFile instances with references\n    // to the linked storage.\n    const { assets } = constructor.definition\n    if (assets) {\n      for (const dataPath in assets) {\n        const storage = constructor.app.getStorage(assets[dataPath].storage)\n        const data = getValueAtDataPath(json, dataPath, () => null)\n        if (data) {\n          const convertToAssetFiles = data => {\n            if (data) {\n              if (isArray(data)) {\n                data.forEach(convertToAssetFiles)\n              } else {\n                storage.convertAssetFile(data)\n              }\n            }\n          }\n          convertToAssetFiles(data)\n        }\n      }\n    }\n    return json\n  }\n\n  // @override\n  $formatJson(json) {\n    const { constructor } = this\n    // Calculate and set the computed properties.\n    for (const key of constructor.computedAttributes) {\n      // Perhaps the computed property is produced in the SQL statement,\n      // in which case we don't have to do anything anymore here.\n      if (!(key in json)) {\n        const value = this[key]\n        if (value !== undefined) {\n          json[key] = value\n        }\n      }\n    }\n    // Remove hidden attributes.\n    for (const key of constructor.hiddenAttributes) {\n      delete json[key]\n    }\n    return json\n  }\n\n  // Graph handling\n\n  $filterGraph(modelGraph, expr) {\n    return filterGraph(this.constructor, modelGraph, expr)\n  }\n\n  async $populateGraph(modelGraph, expr, trx) {\n    return populateGraph(this.constructor, modelGraph, expr, trx)\n  }\n\n  static filterGraph(modelGraph, expr) {\n    return filterGraph(this, modelGraph, expr)\n  }\n\n  static async populateGraph(modelGraph, expr, trx) {\n    return populateGraph(this, modelGraph, expr, trx)\n  }\n\n  static getPropertyOrRelationAtDataPath(dataPath) {\n    // Finds the property or relation at the given the dataPath of the model by\n    // parsing the dataPath and matching it to its relations and properties.\n    const parsedDataPath = parseDataPath(dataPath)\n    let index = 0\n\n    const getResult = (property = null, relation = null) => {\n      const found = property || relation\n      const name = parsedDataPath[index]\n      const next = index + 1\n      const dataPath = found\n        ? normalizeDataPath(parsedDataPath.slice(0, next))\n        : null\n      const nestedDataPath = found\n        ? normalizeDataPath(parsedDataPath.slice(next))\n        : null\n      const expression = found\n        ? parsedDataPath.slice(0, relation ? next : index).join('.') +\n          (property ? `(#${name})` : '')\n        : null\n      return {\n        property,\n        relation,\n        dataPath,\n        nestedDataPath,\n        name,\n        expression,\n        index\n      }\n    }\n\n    const [firstToken, ...otherTokens] = parsedDataPath\n    const property = this.definition.properties[firstToken]\n    if (property) {\n      return getResult(property)\n    } else {\n      let relation = this.getRelations()[firstToken]\n      if (relation) {\n        let { relatedModelClass } = relation\n        for (const token of otherTokens) {\n          index++\n          const property = relatedModelClass.definition.properties[token]\n          if (property) {\n            return getResult(property)\n          } else if (token === '*') {\n            if (relation.isOneToOne()) {\n              // Do not support wildcards on one-to-one relations:\n              return getResult()\n            } else {\n              continue\n            }\n          } else {\n            // Found a relation? Keep iterating.\n            relation = relatedModelClass.getRelations()[token]\n            if (relation) {\n              relatedModelClass = relation.relatedModelClass\n            } else {\n              return getResult()\n            }\n          }\n        }\n        if (relation) {\n          // Still here? Found a relation at the end of the data-path.\n          return getResult(null, relation)\n        }\n      }\n    }\n    return getResult()\n  }\n\n  // @override\n  static relatedQuery(relationName, trx) {\n    // https://github.com/Vincit/objection.js/issues/1720\n    return super.relatedQuery(relationName, trx).alias(relationName)\n  }\n\n  // @override\n  static modifierNotFound(query, modifier) {\n    if (isString(modifier)) {\n      if (query.modelClass().hasScope(modifier)) {\n        return query.applyScope(modifier)\n      }\n      // Now check possible scope prefixes and handle them:\n      switch (modifier[0]) {\n      case '^': // Eager-applied scope:\n        // Always apply eager-scopes, even if the model itself doesn't know it.\n        // The scope may still be known in eager-loaded relations.\n        // Note: `applyScope()` will handle the '^' sign.\n        return query.applyScope(modifier)\n      case '-': // Ignore scope:\n        return query.ignoreScope(modifier.slice(1))\n      case '#': // Select column:\n        return query.select(modifier.slice(1))\n      }\n    }\n    super.modifierNotFound(query, modifier)\n  }\n\n  // @override\n  static createNotFoundError(ctx, error) {\n    return new NotFoundError(\n      error || (\n        ctx.byId\n          ? `'${this.name}' model with id ${ctx.byId} not found`\n          : `'${this.name}' model not found`\n      )\n    )\n  }\n\n  // @override\n  static createValidator() {\n    // Use a shared validator per app, so model schema can reference each other.\n    // NOTE: The Dito Validator class creates and manages this shared Objection\n    // Validator instance for us, we just need to return it here:\n    return this.app.validator\n  }\n\n  // @override\n  static createValidationError({ type, message, errors, options, json }) {\n    switch (type) {\n    case 'ModelValidation':\n      return this.app.createValidationError({\n        type,\n        message:\n          message || `The provided data for the ${this.name} model is not valid`,\n        errors,\n        options,\n        json\n      })\n    case 'RelationExpression':\n    case 'UnallowedRelation':\n      return new RelationError({ type, message, errors })\n    case 'InvalidGraph':\n      return new GraphError({ type, message, errors })\n    default:\n      return new ResponseError({ type, message, errors })\n    }\n  }\n\n  // @override\n  static QueryBuilder = QueryBuilder\n\n  // https://vincit.github.io/objection.js/api/model/static-properties.html#static-cloneobjectattributes\n  static cloneObjectAttributes = false\n\n  // Only pick properties for database JSON that is mentioned in the schema.\n  static pickJsonSchemaProperties = true\n\n  // See https://gitter.im/Vincit/objection.js?at=5a81f859ce68c3bc7479d65a\n  static useLimitInFirst = true\n\n  static get definition() {\n    // Check if we already have a definition object for this class and return it\n    return getMeta(this, 'definition', () => {\n      const definition = {}\n\n      const setDefinition = (name, property) => {\n        Object.defineProperty(definition, name, {\n          ...property,\n          enumerable: true\n        })\n      }\n\n      const getDefinition = name => {\n        let modelClass = this\n        // Collect ancestor values for proper inheritance.\n        // NOTE: values are collected in sequence of inheritance, from sub-class\n        // to super-class. To go from super-class to sub-class when merging,\n        // `mergeReversed()` is used to prevent wrong overrides.\n        // `mergeAsReversedArrays()` can be used to keep arrays of inherited\n        // values per key, see `definitions.hooks`.\n        const values = []\n        while (modelClass !== objection.Model) {\n          // Only consider model classes that actually define `name` property.\n          if (name in modelClass) {\n            // Use reflection through getOwnPropertyDescriptor() to be able to\n            // call the getter on `this` rather than on `modelClass`. This can\n            // be used to provide abstract base-classes and have them create\n            // their relations for `this` inside `get relations()` accessors.\n            const desc = Object.getOwnPropertyDescriptor(modelClass, name)\n            if (desc) {\n              const value = desc.get?.call(this) || desc.value\n              if (value) {\n                values.push(value)\n              }\n            }\n          }\n          modelClass = Object.getPrototypeOf(modelClass)\n        }\n        // To prevent endless recursion with interdependent calls related to\n        // properties, override definition before calling handler():\n        setDefinition(name, {\n          configurable: true,\n          value: {}\n        })\n        try {\n          const merged = definitions[name].call(this, values)\n          // Once calculated, override getter with final merged value.\n          setDefinition(name, {\n            configurable: false,\n            value: merged\n          })\n          return merged\n        } catch (error) {\n          throw new ModelError(this, error.message)\n        }\n      }\n\n      // If no definition object was defined yet, create one with accessors for\n      // each entry in `definitions`. Each of these getters when called merge\n      // definitions up the inheritance chain and store the merged result in\n      // `modelClass.definition[name]` for further caching.\n      for (const name in definitions) {\n        setDefinition(name, {\n          configurable: true,\n          get: () => getDefinition(name)\n        })\n      }\n      return definition\n    })\n  }\n\n  // Hooks\n\n  $emit(event, ...args) {\n    return this.constructor.emit(event, this, ...args)\n  }\n\n  static beforeFind(args) {\n    return this._emitStaticHook('before:find', args)\n  }\n\n  static afterFind(args) {\n    return this._emitStaticHook('after:find', args)\n  }\n\n  static beforeInsert(args) {\n    return this._emitStaticHook('before:insert', args)\n  }\n\n  static afterInsert(args) {\n    return this._emitStaticHook('after:insert', args)\n  }\n\n  static beforeUpdate(args) {\n    return this._emitStaticHook('before:update', args)\n  }\n\n  static afterUpdate(args) {\n    return this._emitStaticHook('after:update', args)\n  }\n\n  static beforeDelete(args) {\n    return this._emitStaticHook('before:delete', args)\n  }\n\n  static afterDelete(args) {\n    return this._emitStaticHook('after:delete', args)\n  }\n\n  static async _emitStaticHook(event, originalArgs) {\n    const listeners = this.listeners(event)\n    if (listeners.length > 0) {\n      // Static hooks are emitted in sequence (but each event can be async), and\n      // results are passed through and returned in the end.\n      let { result } = originalArgs\n      // The result of any event handler will override `args.result` in the call\n      // of the next handler in sequence. As `StaticHookArguments` in Objection\n      // is private, use a JS inheritance trick here to override `args.result`:\n      const args = Object.create(originalArgs, {\n        type: {\n          value: event\n        },\n        result: {\n          get() {\n            return result\n          }\n        }\n      })\n      for (const listener of listeners) {\n        const res = await listener.call(this, args)\n        if (res !== undefined) {\n          result = res\n        }\n      }\n      // Unfortunately `result` is always an array, even when the actual result\n      // is a model object. Avoid returning it when it's not actually changed.\n      // See: https://github.com/Vincit/objection.js/issues/1842\n      if (result !== originalArgs.result) {\n        return result\n      }\n    }\n  }\n\n  // Assets handling\n\n  static _setupAssetsEvents(assets) {\n    const assetDataPaths = Object.keys(assets)\n\n    this.on([\n      'before:insert',\n      'before:update',\n      'before:delete'\n    ], async ({ type, transaction, inputItems, asFindQuery }) => {\n      const afterItems = type === 'before:delete'\n        ? []\n        : inputItems\n      // Figure out which asset data paths where actually present in the\n      // submitted data, and only compare these. But when deleting, use all.\n      const dataPaths = afterItems.length > 0\n        ? assetDataPaths.filter(\n          path => getValueAtAssetDataPath(afterItems[0], path) !== undefined\n        )\n        : assetDataPaths\n\n      // `dataPaths` will be empty in the case of an update/insert that do not\n      // affect the assets.\n      if (dataPaths.length === 0) {\n        return\n      }\n\n      // Load the model's asset files in their current state before the query is\n      // executed.\n      const beforeItems = type === 'before:insert'\n        ? []\n        : await loadAssetDataPaths(asFindQuery(), dataPaths)\n      const beforeFilesPerDataPath = getFilesPerAssetDataPath(\n        beforeItems,\n        dataPaths\n      )\n      const afterFilesPerDataPath = getFilesPerAssetDataPath(\n        afterItems,\n        dataPaths\n      )\n\n      const importedFiles = []\n      const modifiedFiles = []\n\n      if (transaction.rollback) {\n        // Prevent wrong memory leak error messages when installing more than 10\n        // 'rollback' handlers, which can happen with more complex queries.\n        transaction.setMaxListeners(0)\n        transaction.on('rollback', async error => {\n          if (importedFiles.length > 0) {\n            console.info(\n              `Received '${error}', removing imported files again: ${\n                importedFiles.map(file => `'${file.name}'`)\n              }`\n            )\n            await Promise.all(\n              importedFiles.map(\n                file => file.storage.removeFile(file)\n              )\n            )\n          }\n          if (modifiedFiles.length > 0) {\n            // TODO: `modifiedFiles` should be restored as well, but that's far\n            // from trivial since no backup is kept in `handleModifiedAssets`\n            console.info(\n              `Unable to restore these already modified files: ${\n                modifiedFiles.map(file => `'${file.name}'`)\n              }`\n            )\n          }\n        })\n      }\n\n      for (const dataPath of dataPaths) {\n        const storage = this.app.getStorage(assets[dataPath].storage)\n        const beforeFiles = beforeFilesPerDataPath[dataPath] || []\n        const afterFiles = afterFilesPerDataPath[dataPath] || []\n        const beforeByKey = mapFilesByKey(beforeFiles)\n        const afterByKey = mapFilesByKey(afterFiles)\n        const removedFiles = beforeFiles.filter(file => !afterByKey[file.key])\n        const addedFiles = afterFiles.filter(file => !beforeByKey[file.key])\n        // Also handle modified files, which are files where the data property\n        // is changed before update / patch, meanting the file is changed.\n        // NOTE: This will change the content for all the references to it,\n        // and thus should only really be used when there's only one reference.\n        const modifiedFiles = afterFiles.filter(\n          file => file.data && beforeByKey[file.key]\n        )\n        importedFiles.push(\n          ...await this.app.handleAdddedAndRemovedAssets(\n            storage,\n            addedFiles,\n            removedFiles,\n            transaction\n          )\n        )\n        modifiedFiles.push(\n          ...await this.app.handleModifiedAssets(\n            storage,\n            modifiedFiles,\n            transaction\n          )\n        )\n      }\n    })\n  }\n}\n\nEventEmitter.mixin(Model)\nKnexHelper.mixin(Model)\n// Expose a selection of QueryBuilder methods as static methods on model classes\nQueryBuilder.mixin(Model)\n\nconst metaMap = new WeakMap()\n\nfunction getMeta(modelClass, key, value) {\n  let meta = metaMap.get(modelClass)\n  if (!meta) {\n    metaMap.set(modelClass, meta = {})\n  }\n  if (!(key in meta)) {\n    meta[key] = isFunction(value) ? value() : value\n  }\n  return meta[key]\n}\n\nfunction loadAssetDataPaths(query, dataPaths) {\n  return dataPaths.reduce(\n    (query, dataPath) => query.loadDataPath(dataPath),\n    query\n  )\n}\n\nfunction getValueAtAssetDataPath(item, path) {\n  return getValueAtDataPath(item, path, () => undefined)\n}\n\nfunction getFilesPerAssetDataPath(items, dataPaths) {\n  return dataPaths.reduce(\n    (allFiles, dataPath) => {\n      allFiles[dataPath] = asArray(items).reduce(\n        (files, item) => {\n          const data = asArray(getValueAtAssetDataPath(item, dataPath))\n          // Use flatten() as dataPath may contain wildcards, resulting in\n          // nested files arrays.\n          files.push(...flatten(data).filter(file => !!file))\n          return files\n        },\n        []\n      )\n      return allFiles\n    },\n    {}\n  )\n}\n\nfunction mapFilesByKey(files) {\n  return files.reduce(\n    (map, file) => {\n      map[file.key] = file\n      return map\n    },\n    {}\n  )\n}\n"]}