@decaf-ts/decorator-validation 1.7.0 → 1.7.2

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 (50) hide show
  1. package/README.md +312 -20
  2. package/dist/decorator-validation.cjs +245 -94
  3. package/dist/decorator-validation.esm.cjs +245 -95
  4. package/lib/constants/validation.cjs +6 -8
  5. package/lib/constants/validation.d.ts +4 -6
  6. package/lib/esm/constants/validation.d.ts +4 -6
  7. package/lib/esm/constants/validation.js +6 -8
  8. package/lib/esm/index.d.ts +8 -1
  9. package/lib/esm/index.js +9 -2
  10. package/lib/esm/model/Model.d.ts +164 -40
  11. package/lib/esm/model/Model.js +165 -41
  12. package/lib/esm/model/construction.d.ts +4 -2
  13. package/lib/esm/model/construction.js +4 -2
  14. package/lib/esm/model/decorators.d.ts +25 -1
  15. package/lib/esm/model/decorators.js +29 -2
  16. package/lib/esm/model/types.d.ts +48 -31
  17. package/lib/esm/model/types.js +1 -1
  18. package/lib/esm/utils/decorators.d.ts +7 -12
  19. package/lib/esm/utils/decorators.js +8 -13
  20. package/lib/esm/validation/Validators/EmailValidator.js +1 -1
  21. package/lib/esm/validation/Validators/PasswordValidator.js +1 -1
  22. package/lib/esm/validation/Validators/URLValidator.js +1 -1
  23. package/lib/esm/validation/Validators/decorators.d.ts +1 -1
  24. package/lib/esm/validation/Validators/decorators.js +2 -2
  25. package/lib/esm/validation/Validators/utils.d.ts +4 -0
  26. package/lib/esm/validation/Validators/utils.js +5 -1
  27. package/lib/esm/validation/decorators.d.ts +20 -26
  28. package/lib/esm/validation/decorators.js +21 -27
  29. package/lib/index.cjs +9 -2
  30. package/lib/index.d.ts +8 -1
  31. package/lib/model/Model.cjs +165 -41
  32. package/lib/model/Model.d.ts +164 -40
  33. package/lib/model/construction.cjs +4 -2
  34. package/lib/model/construction.d.ts +4 -2
  35. package/lib/model/decorators.cjs +29 -2
  36. package/lib/model/decorators.d.ts +25 -1
  37. package/lib/model/types.cjs +1 -1
  38. package/lib/model/types.d.ts +48 -31
  39. package/lib/utils/decorators.cjs +8 -13
  40. package/lib/utils/decorators.d.ts +7 -12
  41. package/lib/validation/Validators/EmailValidator.cjs +1 -1
  42. package/lib/validation/Validators/PasswordValidator.cjs +1 -1
  43. package/lib/validation/Validators/URLValidator.cjs +1 -1
  44. package/lib/validation/Validators/decorators.cjs +2 -2
  45. package/lib/validation/Validators/decorators.d.ts +1 -1
  46. package/lib/validation/Validators/utils.cjs +5 -1
  47. package/lib/validation/Validators/utils.d.ts +4 -0
  48. package/lib/validation/decorators.cjs +21 -27
  49. package/lib/validation/decorators.d.ts +20 -26
  50. package/package.json +1 -1
@@ -149,44 +149,60 @@ export class Model {
149
149
  // eslint-disable-next-line @typescript-eslint/no-unused-vars
150
150
  constructor(arg) { }
151
151
  /**
152
- * @summary Validates the object according to its decorated properties
152
+ * @description Validates the model object against its defined validation rules
153
+ * @summary Validates the object according to its decorated properties, returning any validation errors
153
154
  *
154
- * @param {any[]} [exceptions] properties in the object to be ignored for the validation. Marked as 'any' to allow for extension but expects strings
155
+ * @param {any[]} [exceptions] - Properties in the object to be ignored for the validation. Marked as 'any' to allow for extension but expects strings
156
+ * @return {ModelErrorDefinition | undefined} - Returns a ModelErrorDefinition object if validation errors exist, otherwise undefined
155
157
  */
156
158
  hasErrors(...exceptions) {
157
159
  return validate(this, ...exceptions);
158
160
  }
159
161
  /**
160
- * @summary Compare object equality recursively
161
- * @param {any} obj object to compare to
162
- * @param {string} [exceptions] property names to be excluded from the comparison
162
+ * @description Determines if this model is equal to another object
163
+ * @summary Compare object equality recursively, checking all properties unless excluded
164
+ *
165
+ * @param {any} obj - Object to compare to
166
+ * @param {string[]} [exceptions] - Property names to be excluded from the comparison
167
+ * @return {boolean} - True if objects are equal, false otherwise
163
168
  */
164
169
  equals(obj, ...exceptions) {
165
170
  return isEqual(this, obj, ...exceptions);
166
171
  }
167
172
  /**
173
+ * @description Converts the model to a serialized string representation
168
174
  * @summary Returns the serialized model according to the currently defined {@link Serializer}
175
+ *
176
+ * @return {string} - The serialized string representation of the model
169
177
  */
170
178
  serialize() {
171
179
  return Model.serialize(this);
172
180
  }
173
181
  /**
174
- * @summary Override the implementation for js's 'toString()' which sucks...
182
+ * @description Provides a human-readable string representation of the model
183
+ * @summary Override the implementation for js's 'toString()' to provide a more useful representation
184
+ *
185
+ * @return {string} - A string representation of the model including its class name and JSON representation
175
186
  * @override
176
187
  */
177
188
  toString() {
178
189
  return this.constructor.name + ": " + JSON.stringify(this, undefined, 2);
179
190
  }
180
191
  /**
181
- * @summary Defines a default implementation for object hash. Relies on a very basic implementation based on Java's string hash;
192
+ * @description Generates a hash string for the model object
193
+ * @summary Defines a default implementation for object hash, relying on a basic implementation based on Java's string hash
194
+ *
195
+ * @return {string} - A hash string representing the model
182
196
  */
183
197
  hash() {
184
198
  return Model.hash(this);
185
199
  }
186
200
  /**
187
- * @summary Deserializes a Model
188
- * @param {string} str
201
+ * @description Converts a serialized string back into a model instance
202
+ * @summary Deserializes a Model from its string representation
189
203
  *
204
+ * @param {string} str - The serialized string to convert back to a model
205
+ * @return {any} - The deserialized model instance
190
206
  * @throws {Error} If it fails to parse the string, or if it fails to build the model
191
207
  */
192
208
  static deserialize(str) {
@@ -196,12 +212,13 @@ export class Model {
196
212
  return Serialization.deserialize(str);
197
213
  }
198
214
  /**
215
+ * @description Copies properties from a source object to a model instance
199
216
  * @summary Repopulates the Object properties with the ones from the new object
200
- * @description Iterates all common properties of obj (if existing) and self, and copies them onto self
201
- *
202
- * @param {T} self
203
- * @param {T | Record<string, any>} [obj]
204
217
  *
218
+ * @template T
219
+ * @param {T} self - The target model instance to update
220
+ * @param {T | Record<string, any>} [obj] - The source object containing properties to copy
221
+ * @return {T} - The updated model instance
205
222
  */
206
223
  static fromObject(self, obj) {
207
224
  if (!obj)
@@ -212,14 +229,46 @@ export class Model {
212
229
  return self;
213
230
  }
214
231
  /**
215
- * @summary Repopulates the instance with the ones from the new Model Object
216
- * @description Iterates all common properties of obj (if existing) and self, and copies them onto self.
217
- * Is aware of nested Model Objects and rebuilds them also.
218
- * When List properties are decorated with {@link list}, they list items will also be rebuilt
232
+ * @description Copies and rebuilds properties from a source object to a model instance, handling nested models
233
+ * @summary Repopulates the instance with properties from the new Model Object, recursively rebuilding nested models
219
234
  *
220
- * @param {T} self
221
- * @param {T | Record<string, any>} [obj]
235
+ * @template T
236
+ * @param {T} self - The target model instance to update
237
+ * @param {T | Record<string, any>} [obj] - The source object containing properties to copy
238
+ * @return {T} - The updated model instance with rebuilt nested models
222
239
  *
240
+ * @mermaid
241
+ * sequenceDiagram
242
+ * participant C as Client
243
+ * participant M as Model.fromModel
244
+ * participant B as Model.build
245
+ * participant R as Reflection
246
+ *
247
+ * C->>M: fromModel(self, obj)
248
+ * M->>M: Get attributes from self
249
+ * loop For each property
250
+ * M->>M: Copy property from obj to self
251
+ * alt Property is a model
252
+ * M->>M: Check if property is a model
253
+ * M->>B: build(property, modelType)
254
+ * B-->>M: Return built model
255
+ * else Property is a complex type
256
+ * M->>R: Get property decorators
257
+ * R-->>M: Return decorators
258
+ * M->>M: Filter type decorators
259
+ * alt Property is Array/Set with list decorator
260
+ * M->>M: Process each item in collection
261
+ * loop For each item
262
+ * M->>B: build(item, itemModelType)
263
+ * B-->>M: Return built model
264
+ * end
265
+ * else Property is another model type
266
+ * M->>B: build(property, propertyType)
267
+ * B-->>M: Return built model
268
+ * end
269
+ * end
270
+ * end
271
+ * M-->>C: Return updated self
223
272
  */
224
273
  static fromModel(self, obj) {
225
274
  if (!obj)
@@ -299,22 +348,30 @@ export class Model {
299
348
  return self;
300
349
  }
301
350
  /**
302
- * @summary Sets the Global {@link ModelBuilderFunction}
303
- * @param {ModelBuilderFunction} [builder]
351
+ * @description Configures the global model builder function
352
+ * @summary Sets the Global {@link ModelBuilderFunction} used for building model instances
353
+ *
354
+ * @param {ModelBuilderFunction} [builder] - The builder function to set as the global builder
355
+ * @return {void}
304
356
  */
305
357
  static setBuilder(builder) {
306
358
  modelBuilderFunction = builder;
307
359
  }
308
360
  /**
309
- * @summary Retrieves the current global {@link ModelBuilderFunction}
361
+ * @description Retrieves the currently configured global model builder function
362
+ * @summary Returns the current global {@link ModelBuilderFunction} used for building model instances
363
+ *
364
+ * @return {ModelBuilderFunction | undefined} - The current global builder function or undefined if not set
310
365
  */
311
366
  static getBuilder() {
312
367
  return modelBuilderFunction;
313
368
  }
314
369
  /**
315
- * Returns the current {@link ModelRegistryManager}
370
+ * @description Provides access to the current model registry
371
+ * @summary Returns the current {@link ModelRegistryManager} instance, creating one if it doesn't exist
316
372
  *
317
- * @return ModelRegistry, defaults to {@link ModelRegistryManager}
373
+ * @return {ModelRegistry<any>} - The current model registry, defaults to a new {@link ModelRegistryManager} if not set
374
+ * @private
318
375
  */
319
376
  static getRegistry() {
320
377
  if (!actingModelRegistry)
@@ -322,17 +379,23 @@ export class Model {
322
379
  return actingModelRegistry;
323
380
  }
324
381
  /**
325
- * Returns the current actingModelRegistry
382
+ * @description Configures the model registry to be used by the Model system
383
+ * @summary Sets the current model registry to a custom implementation
326
384
  *
327
- * @param {BuilderRegistry} modelRegistry the new implementation of Registry
385
+ * @param {BuilderRegistry<any>} modelRegistry - The new implementation of Registry to use
386
+ * @return {void}
328
387
  */
329
388
  static setRegistry(modelRegistry) {
330
389
  actingModelRegistry = modelRegistry;
331
390
  }
332
391
  /**
333
- * @summary register new Models
334
- * @param {any} constructor
335
- * @param {string} [name] when not defined, the name of the constructor will be used
392
+ * @description Registers a model constructor with the model registry
393
+ * @summary Registers new model classes to make them available for serialization and deserialization
394
+ *
395
+ * @template T
396
+ * @param {ModelConstructor<T>} constructor - The model constructor to register
397
+ * @param {string} [name] - Optional name to register the constructor under, defaults to constructor.name
398
+ * @return {void}
336
399
  *
337
400
  * @see ModelRegistry
338
401
  */
@@ -340,8 +403,12 @@ export class Model {
340
403
  return Model.getRegistry().register(constructor, name);
341
404
  }
342
405
  /**
343
- * @summary Gets a registered Model {@link ModelConstructor}
344
- * @param {string} name
406
+ * @description Retrieves a registered model constructor by name
407
+ * @summary Gets a registered Model {@link ModelConstructor} from the model registry
408
+ *
409
+ * @template T
410
+ * @param {string} name - The name of the model constructor to retrieve
411
+ * @return {ModelConstructor<T> | undefined} - The model constructor if found, undefined otherwise
345
412
  *
346
413
  * @see ModelRegistry
347
414
  */
@@ -349,19 +416,39 @@ export class Model {
349
416
  return Model.getRegistry().get(name);
350
417
  }
351
418
  /**
352
- * @param {Record<string, any>} obj
353
- * @param {string} [clazz] when provided, it will attempt to find the matching constructor
419
+ * @description Creates a model instance from a plain object
420
+ * @summary Builds a model instance using the model registry, optionally specifying the model class
354
421
  *
355
- * @throws Error If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property
422
+ * @template T
423
+ * @param {Record<string, any>} obj - The source object to build the model from
424
+ * @param {string} [clazz] - When provided, it will attempt to find the matching constructor by name
425
+ * @return {T} - The built model instance
426
+ * @throws {Error} If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property
356
427
  *
357
428
  * @see ModelRegistry
358
429
  */
359
430
  static build(obj = {}, clazz) {
360
431
  return Model.getRegistry().build(obj, clazz);
361
432
  }
433
+ /**
434
+ * @description Retrieves the model metadata from a model instance
435
+ * @summary Gets the metadata associated with a model instance, typically the model class name
436
+ *
437
+ * @template M
438
+ * @param {M} model - The model instance to get metadata from
439
+ * @return {string} - The model metadata (typically the class name)
440
+ */
362
441
  static getMetadata(model) {
363
442
  return getMetadata(model);
364
443
  }
444
+ /**
445
+ * @description Retrieves all attribute names from a model class or instance
446
+ * @summary Gets all attributes defined in a model, traversing the prototype chain to include inherited attributes
447
+ *
448
+ * @template V
449
+ * @param {Constructor<V> | V} model - The model class or instance to get attributes from
450
+ * @return {string[]} - Array of attribute names defined in the model
451
+ */
365
452
  static getAttributes(model) {
366
453
  const result = [];
367
454
  let prototype = model instanceof Model
@@ -376,18 +463,53 @@ export class Model {
376
463
  }
377
464
  return result;
378
465
  }
466
+ /**
467
+ * @description Compares two model instances for equality
468
+ * @summary Determines if two model instances are equal by comparing their properties
469
+ *
470
+ * @template M
471
+ * @param {M} obj1 - First model instance to compare
472
+ * @param {M} obj2 - Second model instance to compare
473
+ * @param {any[]} [exceptions] - Property names to exclude from comparison
474
+ * @return {boolean} - True if the models are equal, false otherwise
475
+ */
379
476
  static equals(obj1, obj2, ...exceptions) {
380
477
  return isEqual(obj1, obj2, ...exceptions);
381
478
  }
479
+ /**
480
+ * @description Validates a model instance against its validation rules
481
+ * @summary Checks if a model has validation errors, optionally ignoring specified properties
482
+ *
483
+ * @template M
484
+ * @param {M} model - The model instance to validate
485
+ * @param {string[]} [propsToIgnore] - Properties to exclude from validation
486
+ * @return {ModelErrorDefinition | undefined} - Returns validation errors if any, otherwise undefined
487
+ */
382
488
  static hasErrors(model, ...propsToIgnore) {
383
489
  return validate(model, ...propsToIgnore);
384
490
  }
491
+ /**
492
+ * @description Converts a model instance to a serialized string
493
+ * @summary Serializes a model instance using the configured serializer or the default one
494
+ *
495
+ * @template M
496
+ * @param {M} model - The model instance to serialize
497
+ * @return {string} - The serialized string representation of the model
498
+ */
385
499
  static serialize(model) {
386
500
  const metadata = Reflect.getMetadata(Model.key(ModelKeys.SERIALIZATION), model.constructor);
387
501
  if (metadata && metadata.serializer)
388
502
  return Serialization.serialize(this, metadata.serializer, ...(metadata.args || []));
389
503
  return Serialization.serialize(model);
390
504
  }
505
+ /**
506
+ * @description Generates a hash string for a model instance
507
+ * @summary Creates a hash representation of a model using the configured algorithm or the default one
508
+ *
509
+ * @template M
510
+ * @param {M} model - The model instance to hash
511
+ * @return {string} - The hash string representing the model
512
+ */
391
513
  static hash(model) {
392
514
  const metadata = Reflect.getMetadata(Model.key(ModelKeys.HASHING), model.constructor);
393
515
  if (metadata && metadata.algorithm)
@@ -395,9 +517,11 @@ export class Model {
395
517
  return Hashing.hash(model);
396
518
  }
397
519
  /**
520
+ * @description Creates a metadata key for use with the Reflection API
398
521
  * @summary Builds the key to store as Metadata under Reflections
399
- * @description concatenates {@link ModelKeys#REFLECT} with the provided key
400
- * @param {string} str
522
+ *
523
+ * @param {string} str - The base key to concatenate with the model reflection prefix
524
+ * @return {string} - The complete metadata key
401
525
  */
402
526
  static key(str) {
403
527
  return getModelKey(str);
@@ -435,12 +559,12 @@ export class Model {
435
559
  /**
436
560
  * @description Checks if a property of a model is itself a model or has a model type
437
561
  * @summary Determines whether a specific property of a model instance is either a model instance
438
- * or has a type that is registered as a model. This function is used for model serialization
439
- * and deserialization to properly handle nested models.
440
- * @template M extends {@link Model}
562
+ * or has a type that is registered as a model
563
+ *
564
+ * @template M
441
565
  * @param {M} target - The model instance to check
442
566
  * @param {string} attribute - The property name to check
443
- * @return {boolean | string | undefined} Returns true if the property is a model instance,
567
+ * @return {boolean | string | undefined} - Returns true if the property is a model instance,
444
568
  * the model name if the property has a model type, or undefined if not a model
445
569
  */
446
570
  static isPropertyModel(target, attribute) {
@@ -450,4 +574,4 @@ export class Model {
450
574
  return Model.get(metadata.name) ? metadata.name : undefined;
451
575
  }
452
576
  }
453
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Model.js","sourceRoot":"","sources":["../../../src/model/Model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Serialization } from \"../utils/serialization\";\nimport { BuilderRegistry } from \"../utils/registry\";\nimport { ModelErrorDefinition } from \"./ModelErrorDefinition\";\nimport {\n  Comparable,\n  Constructor,\n  Hashable,\n  ModelArg,\n  ModelBuilderFunction,\n  ModelConstructor,\n  Serializable,\n  Validatable,\n} from \"./types\";\nimport { DecoratorMetadata, isEqual, Reflection } from \"@decaf-ts/reflection\";\nimport { validate } from \"./validation\";\nimport { Hashing } from \"../utils/hashing\";\nimport { ModelKeys } from \"../utils/constants\";\nimport { ValidationKeys } from \"../validation/Validators/constants\";\nimport { jsTypes, ReservedModels } from \"./constants\";\nimport { getModelKey, getMetadata } from \"./utils\";\n\nlet modelBuilderFunction: ModelBuilderFunction | undefined;\nlet actingModelRegistry: BuilderRegistry<any>;\n\n/**\n * @description Registry type for storing and retrieving model constructors\n * @summary The ModelRegistry type defines a registry for model constructors that extends\n * the BuilderRegistry interface. It provides a standardized way to register, retrieve,\n * and build model instances, enabling the model system to work with different types of models.\n *\n * @interface ModelRegistry\n * @template T Type of model that can be registered, must extend Model\n * @extends BuilderRegistry<T>\n * @memberOf module:decorator-validation\n * @category Model\n */\nexport type ModelRegistry<T extends Model> = BuilderRegistry<T>;\n\n/**\n * @description Registry manager for model constructors that enables serialization and rebuilding\n * @summary The ModelRegistryManager implements the ModelRegistry interface and provides\n * functionality for registering, retrieving, and building model instances. It maintains\n * a cache of model constructors indexed by name, allowing for efficient lookup and instantiation.\n * This class is essential for the serialization and deserialization of model objects.\n *\n * @param {function(Record<string, any>): boolean} [testFunction] - Function to test if an object is a model, defaults to {@link Model#isModel}\n *\n * @class ModelRegistryManager\n * @template M Type of model that can be registered, must extend Model\n * @implements ModelRegistry<M>\n * @category Model\n *\n * @example\n * ```typescript\n * // Create a model registry\n * const registry = new ModelRegistryManager();\n *\n * // Register a model class\n * registry.register(User);\n *\n * // Retrieve a model constructor by name\n * const UserClass = registry.get(\"User\");\n *\n * // Build a model instance from a plain object\n * const userData = { name: \"John\", age: 30 };\n * const user = registry.build(userData, \"User\");\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant R as ModelRegistryManager\n *   participant M as Model Class\n *\n *   C->>R: new ModelRegistryManager(testFunction)\n *   C->>R: register(ModelClass)\n *   R->>R: Store in cache\n *   C->>R: get(\"ModelName\")\n *   R-->>C: ModelClass constructor\n *   C->>R: build(data, \"ModelName\")\n *   R->>R: Get constructor from cache\n *   R->>M: new ModelClass(data)\n *   M-->>R: Model instance\n *   R-->>C: Model instance\n */\nexport class ModelRegistryManager<M extends Model> implements ModelRegistry<M> {\n  private cache: Record<string, ModelConstructor<M>> = {};\n  private readonly testFunction: (obj: object) => boolean;\n\n  constructor(\n    testFunction: (obj: Record<string, any>) => boolean = Model.isModel\n  ) {\n    this.testFunction = testFunction;\n  }\n\n  /**\n   * @description Registers a model constructor with the registry\n   * @summary Adds a model constructor to the registry cache, making it available for\n   * later retrieval and instantiation. If no name is provided, the constructor's name\n   * property is used as the key in the registry.\n   *\n   * @param {ModelConstructor<M>} constructor - The model constructor to register\n   * @param {string} [name] - Optional name to register the constructor under, defaults to constructor.name\n   * @return {void}\n   * @throws {Error} If the constructor is not a function\n   */\n  register(constructor: ModelConstructor<M>, name?: string): void {\n    if (typeof constructor !== \"function\")\n      throw new Error(\n        \"Model registering failed. Missing Class name or constructor\"\n      );\n    name = name || constructor.name;\n    this.cache[name] = constructor;\n  }\n\n  /**\n   * @summary Gets a registered Model {@link ModelConstructor}\n   * @param {string} name\n   */\n  get(name: string): ModelConstructor<M> | undefined {\n    try {\n      return this.cache[name];\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: any) {\n      return undefined;\n    }\n  }\n\n  /**\n   * @param {Record<string, any>} obj\n   * @param {string} [clazz] when provided, it will attempt to find the matching constructor\n   *\n   * @throws Error If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property\n   */\n  build(obj: Record<string, any> = {}, clazz?: string): M {\n    if (!clazz && !this.testFunction(obj))\n      throw new Error(\"Provided obj is not a Model object\");\n    const name = clazz || Model.getMetadata(obj as any);\n    if (!(name in this.cache))\n      throw new Error(\n        `Provided class ${name} is not a registered Model object`\n      );\n    return new this.cache[name](obj);\n  }\n}\n\n/**\n * @summary Bulk Registers Models\n * @description Useful when using bundlers that might not evaluate all the code at once\n *\n * @template M extends Model\n * @param {Array<Constructor<M>> | Array<{name: string, constructor: Constructor<M>}>} [models]\n *\n * @memberOf module:decorator-validation\n * @category Model\n */\nexport function bulkModelRegister<M extends Model>(\n  ...models: (Constructor<M> | { name: string; constructor: Constructor<M> })[]\n) {\n  models.forEach(\n    (m: Constructor<M> | { name: string; constructor: Constructor<M> }) => {\n      const constructor: Constructor<M> = (\n        m.constructor ? m.constructor : m\n      ) as Constructor<M>;\n      Model.register(constructor, (m as Constructor<M>).name);\n    }\n  );\n}\n\n/**\n * @summary Abstract class representing a Validatable Model object\n * @description Meant to be used as a base class for all Model classes\n *\n * Model objects must:\n *  - Have all their required properties marked with '!';\n *  - Have all their optional properties marked as '?':\n *\n * @param {ModelArg<Model>} model base object from which to populate properties from\n *\n * @class Model\n * @category Model\n * @abstract\n * @implements Validatable\n * @implements Serializable\n *\n * @example\n *      class ClassName {\n *          @required()\n *          requiredPropertyName!: PropertyType;\n *\n *          optionalPropertyName?: PropertyType;\n *      }\n */\nexport abstract class Model\n  implements Validatable, Serializable, Hashable, Comparable<Model>\n{\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  protected constructor(arg?: ModelArg<Model>) {}\n\n  /**\n   * @summary Validates the object according to its decorated properties\n   *\n   * @param {any[]} [exceptions] properties in the object to be ignored for the validation. Marked as 'any' to allow for extension but expects strings\n   */\n  public hasErrors(...exceptions: any[]): ModelErrorDefinition | undefined {\n    return validate(this, ...exceptions);\n  }\n\n  /**\n   * @summary Compare object equality recursively\n   * @param {any} obj object to compare to\n   * @param {string} [exceptions] property names to be excluded from the comparison\n   */\n  public equals(obj: any, ...exceptions: string[]): boolean {\n    return isEqual(this, obj, ...exceptions);\n  }\n\n  /**\n   * @summary Returns the serialized model according to the currently defined {@link Serializer}\n   */\n  serialize(): string {\n    return Model.serialize(this);\n  }\n\n  /**\n   * @summary Override the implementation for js's 'toString()' which sucks...\n   * @override\n   */\n  public toString(): string {\n    return this.constructor.name + \": \" + JSON.stringify(this, undefined, 2);\n  }\n\n  /**\n   * @summary Defines a default implementation for object hash. Relies on a very basic implementation based on Java's string hash;\n   */\n  public hash(): string {\n    return Model.hash(this);\n  }\n\n  /**\n   * @summary Deserializes a Model\n   * @param {string} str\n   *\n   * @throws {Error} If it fails to parse the string, or if it fails to build the model\n   */\n  static deserialize(str: string) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.SERIALIZATION),\n      this.constructor\n    );\n\n    if (metadata && metadata.serializer)\n      return Serialization.deserialize(\n        str,\n        metadata.serializer,\n        ...(metadata.args || [])\n      );\n    return Serialization.deserialize(str);\n  }\n\n  /**\n   * @summary Repopulates the Object properties with the ones from the new object\n   * @description Iterates all common properties of obj (if existing) and self, and copies them onto self\n   *\n   * @param {T} self\n   * @param {T | Record<string, any>} [obj]\n   *\n   */\n  static fromObject<T extends Model>(\n    self: T,\n    obj?: T | Record<string, any>\n  ): T {\n    if (!obj) obj = {};\n    for (const prop of Model.getAttributes(self)) {\n      (self as any)[prop] = (obj as any)[prop] || undefined;\n    }\n    return self;\n  }\n\n  /**\n   * @summary Repopulates the instance with the ones from the new Model Object\n   * @description Iterates all common properties of obj (if existing) and self, and copies them onto self.\n   * Is aware of nested Model Objects and rebuilds them also.\n   * When List properties are decorated with {@link list}, they list items will also be rebuilt\n   *\n   * @param {T} self\n   * @param {T | Record<string, any>} [obj]\n   *\n   */\n  static fromModel<T extends Model>(self: T, obj?: T | Record<string, any>): T {\n    if (!obj) obj = {};\n\n    let decorators: DecoratorMetadata[], dec: DecoratorMetadata;\n\n    const props = Model.getAttributes(self);\n\n    for (const prop of props) {\n      (self as Record<string, any>)[prop] =\n        (obj as Record<string, any>)[prop] ?? undefined;\n      if (typeof (self as any)[prop] !== \"object\") continue;\n      const propM = Model.isPropertyModel(self, prop);\n      if (propM) {\n        try {\n          (self as Record<string, any>)[prop] = Model.build(\n            (self as Record<string, any>)[prop],\n            typeof propM === \"string\" ? propM : undefined\n          );\n        } catch (e: any) {\n          console.log(e);\n        }\n        continue;\n      }\n\n      const allDecorators: DecoratorMetadata[] =\n        Reflection.getPropertyDecorators(\n          ValidationKeys.REFLECT,\n          self,\n          prop\n        ).decorators;\n      decorators = allDecorators.filter(\n        (d: DecoratorMetadata) =>\n          [ModelKeys.TYPE, ValidationKeys.TYPE as string].indexOf(d.key) !== -1\n      );\n      if (!decorators || !decorators.length)\n        throw new Error(`failed to find decorators for property ${prop}`);\n      dec = decorators.pop() as DecoratorMetadata;\n      const clazz = dec.props.name\n        ? [dec.props.name]\n        : Array.isArray(dec.props.customTypes)\n          ? dec.props.customTypes\n          : [dec.props.customTypes];\n      const reserved = Object.values(ReservedModels).map((v) =>\n        v.toLowerCase()\n      ) as string[];\n\n      clazz.forEach((c) => {\n        if (reserved.indexOf(c.toLowerCase()) === -1)\n          try {\n            switch (c) {\n              case \"Array\":\n              case \"Set\":\n                if (allDecorators.length) {\n                  const listDec = allDecorators.find(\n                    (d) => d.key === ValidationKeys.LIST\n                  );\n                  if (listDec) {\n                    const clazzName = (listDec.props.clazz as string[]).find(\n                      (t: string) => !jsTypes.includes(t.toLowerCase())\n                    );\n                    if (c === \"Array\")\n                      (self as Record<string, any>)[prop] = (\n                        self as Record<string, any>\n                      )[prop].map((el: any) => {\n                        return [\"object\", \"function\"].includes(typeof el) &&\n                          clazzName\n                          ? Model.build(el, clazzName)\n                          : el;\n                      });\n                    if (c === \"Set\") {\n                      const s = new Set();\n                      for (const v of (self as Record<string, any>)[prop]) {\n                        if (\n                          [\"object\", \"function\"].includes(typeof v) &&\n                          clazzName\n                        ) {\n                          s.add(Model.build(v, clazzName));\n                        } else {\n                          s.add(v);\n                        }\n                      }\n                      (self as Record<string, any>)[prop] = s;\n                    }\n                  }\n                }\n                break;\n              default:\n                if ((self as Record<string, any>)[prop])\n                  (self as Record<string, any>)[prop] = Model.build(\n                    (self as any)[prop],\n                    c\n                  );\n            }\n          } catch (e: any) {\n            console.log(e);\n            // do nothing. we have no registry of this class\n          }\n      });\n    }\n    return self;\n  }\n\n  /**\n   * @summary Sets the Global {@link ModelBuilderFunction}\n   * @param {ModelBuilderFunction} [builder]\n   */\n  static setBuilder(builder?: ModelBuilderFunction) {\n    modelBuilderFunction = builder;\n  }\n\n  /**\n   * @summary Retrieves the current global {@link ModelBuilderFunction}\n   */\n  static getBuilder(): ModelBuilderFunction | undefined {\n    return modelBuilderFunction;\n  }\n\n  /**\n   * Returns the current {@link ModelRegistryManager}\n   *\n   * @return ModelRegistry, defaults to {@link ModelRegistryManager}\n   */\n  private static getRegistry() {\n    if (!actingModelRegistry) actingModelRegistry = new ModelRegistryManager();\n    return actingModelRegistry;\n  }\n\n  /**\n   * Returns the current actingModelRegistry\n   *\n   * @param {BuilderRegistry} modelRegistry the new implementation of Registry\n   */\n  static setRegistry(modelRegistry: BuilderRegistry<any>) {\n    actingModelRegistry = modelRegistry;\n  }\n\n  /**\n   * @summary register new Models\n   * @param {any} constructor\n   * @param {string} [name] when not defined, the name of the constructor will be used\n   *\n   * @see ModelRegistry\n   */\n  static register<T extends Model>(\n    constructor: ModelConstructor<T>,\n    name?: string\n  ): void {\n    return Model.getRegistry().register(constructor, name);\n  }\n\n  /**\n   * @summary Gets a registered Model {@link ModelConstructor}\n   * @param {string} name\n   *\n   * @see ModelRegistry\n   */\n  static get<T extends Model>(name: string): ModelConstructor<T> | undefined {\n    return Model.getRegistry().get(name);\n  }\n\n  /**\n   * @param {Record<string, any>} obj\n   * @param {string} [clazz] when provided, it will attempt to find the matching constructor\n   *\n   * @throws Error If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property\n   *\n   * @see ModelRegistry\n   */\n  static build<T extends Model>(\n    obj: Record<string, any> = {},\n    clazz?: string\n  ): T {\n    return Model.getRegistry().build(obj, clazz);\n  }\n\n  static getMetadata<M extends Model>(model: M) {\n    return getMetadata<M>(model);\n  }\n\n  static getAttributes<V extends Model>(model: Constructor<V> | V) {\n    const result: string[] = [];\n    let prototype =\n      model instanceof Model\n        ? Object.getPrototypeOf(model)\n        : (model as any).prototype;\n    while (prototype != null) {\n      const props: string[] = prototype[ModelKeys.ATTRIBUTE];\n      if (props) {\n        result.push(...props);\n      }\n      prototype = Object.getPrototypeOf(prototype);\n    }\n    return result;\n  }\n\n  static equals<M extends Model>(obj1: M, obj2: M, ...exceptions: any[]) {\n    return isEqual(obj1, obj2, ...exceptions);\n  }\n\n  static hasErrors<M extends Model>(model: M, ...propsToIgnore: string[]) {\n    return validate(model, ...propsToIgnore);\n  }\n\n  static serialize<M extends Model>(model: M) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.SERIALIZATION),\n      model.constructor\n    );\n\n    if (metadata && metadata.serializer)\n      return Serialization.serialize(\n        this,\n        metadata.serializer,\n        ...(metadata.args || [])\n      );\n    return Serialization.serialize(model);\n  }\n\n  static hash<M extends Model>(model: M) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.HASHING),\n      model.constructor\n    );\n\n    if (metadata && metadata.algorithm)\n      return Hashing.hash(model, metadata.algorithm, ...(metadata.args || []));\n    return Hashing.hash(model);\n  }\n  /**\n   * @summary Builds the key to store as Metadata under Reflections\n   * @description concatenates {@link ModelKeys#REFLECT} with the provided key\n   * @param {string} str\n   */\n  static key(str: string) {\n    return getModelKey(str);\n  }\n\n  /**\n   * @description Determines if an object is a model instance or has model metadata\n   * @summary Checks whether a given object is either an instance of the Model class or\n   * has model metadata attached to it. This function is essential for serialization and\n   * deserialization processes, as it helps identify model objects that need special handling.\n   * It safely handles potential errors during metadata retrieval.\n   *\n   * @param {Record<string, any>} target - The object to check\n   * @return {boolean} True if the object is a model instance or has model metadata, false otherwise\n   *\n   * @example\n   * ```typescript\n   * // Check if an object is a model\n   * const user = new User({ name: \"John\" });\n   * const isUserModel = isModel(user); // true\n   *\n   * // Check a plain object\n   * const plainObject = { name: \"John\" };\n   * const isPlainObjectModel = isModel(plainObject); // false\n   * ```\n   */\n  static isModel(target: Record<string, any>) {\n    try {\n      return target instanceof Model || !!Model.getMetadata(target as any);\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: any) {\n      return false;\n    }\n  }\n\n  /**\n   * @description Checks if a property of a model is itself a model or has a model type\n   * @summary Determines whether a specific property of a model instance is either a model instance\n   * or has a type that is registered as a model. This function is used for model serialization\n   * and deserialization to properly handle nested models.\n   * @template M extends {@link Model}\n   * @param {M} target - The model instance to check\n   * @param {string} attribute - The property name to check\n   * @return {boolean | string | undefined} Returns true if the property is a model instance,\n   * the model name if the property has a model type, or undefined if not a model\n   */\n  static isPropertyModel<M extends Model>(\n    target: M,\n    attribute: string\n  ): boolean | string | undefined {\n    if (Model.isModel((target as Record<string, any>)[attribute])) return true;\n    const metadata = Reflect.getMetadata(ModelKeys.TYPE, target, attribute);\n    return Model.get(metadata.name) ? metadata.name : undefined;\n  }\n}\n"]}
577
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Model.js","sourceRoot":"","sources":["../../../src/model/Model.ts"],"names":[],"mappings":"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","sourcesContent":["import { Serialization } from \"../utils/serialization\";\nimport { BuilderRegistry } from \"../utils/registry\";\nimport { ModelErrorDefinition } from \"./ModelErrorDefinition\";\nimport {\n  Comparable,\n  Constructor,\n  Hashable,\n  ModelArg,\n  ModelBuilderFunction,\n  ModelConstructor,\n  Serializable,\n  Validatable,\n} from \"./types\";\nimport { DecoratorMetadata, isEqual, Reflection } from \"@decaf-ts/reflection\";\nimport { validate } from \"./validation\";\nimport { Hashing } from \"../utils/hashing\";\nimport { ModelKeys } from \"../utils/constants\";\nimport { ValidationKeys } from \"../validation/Validators/constants\";\nimport { jsTypes, ReservedModels } from \"./constants\";\nimport { getModelKey, getMetadata } from \"./utils\";\n\nlet modelBuilderFunction: ModelBuilderFunction | undefined;\nlet actingModelRegistry: BuilderRegistry<any>;\n\n/**\n * @description Registry type for storing and retrieving model constructors\n * @summary The ModelRegistry type defines a registry for model constructors that extends\n * the BuilderRegistry interface. It provides a standardized way to register, retrieve,\n * and build model instances, enabling the model system to work with different types of models.\n *\n * @interface ModelRegistry\n * @template T Type of model that can be registered, must extend Model\n * @extends BuilderRegistry<T>\n * @memberOf module:decorator-validation\n * @category Model\n */\nexport type ModelRegistry<T extends Model> = BuilderRegistry<T>;\n\n/**\n * @description Registry manager for model constructors that enables serialization and rebuilding\n * @summary The ModelRegistryManager implements the ModelRegistry interface and provides\n * functionality for registering, retrieving, and building model instances. It maintains\n * a cache of model constructors indexed by name, allowing for efficient lookup and instantiation.\n * This class is essential for the serialization and deserialization of model objects.\n *\n * @param {function(Record<string, any>): boolean} [testFunction] - Function to test if an object is a model, defaults to {@link Model#isModel}\n *\n * @class ModelRegistryManager\n * @template M Type of model that can be registered, must extend Model\n * @implements ModelRegistry<M>\n * @category Model\n *\n * @example\n * ```typescript\n * // Create a model registry\n * const registry = new ModelRegistryManager();\n *\n * // Register a model class\n * registry.register(User);\n *\n * // Retrieve a model constructor by name\n * const UserClass = registry.get(\"User\");\n *\n * // Build a model instance from a plain object\n * const userData = { name: \"John\", age: 30 };\n * const user = registry.build(userData, \"User\");\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant R as ModelRegistryManager\n *   participant M as Model Class\n *\n *   C->>R: new ModelRegistryManager(testFunction)\n *   C->>R: register(ModelClass)\n *   R->>R: Store in cache\n *   C->>R: get(\"ModelName\")\n *   R-->>C: ModelClass constructor\n *   C->>R: build(data, \"ModelName\")\n *   R->>R: Get constructor from cache\n *   R->>M: new ModelClass(data)\n *   M-->>R: Model instance\n *   R-->>C: Model instance\n */\nexport class ModelRegistryManager<M extends Model> implements ModelRegistry<M> {\n  private cache: Record<string, ModelConstructor<M>> = {};\n  private readonly testFunction: (obj: object) => boolean;\n\n  constructor(\n    testFunction: (obj: Record<string, any>) => boolean = Model.isModel\n  ) {\n    this.testFunction = testFunction;\n  }\n\n  /**\n   * @description Registers a model constructor with the registry\n   * @summary Adds a model constructor to the registry cache, making it available for\n   * later retrieval and instantiation. If no name is provided, the constructor's name\n   * property is used as the key in the registry.\n   *\n   * @param {ModelConstructor<M>} constructor - The model constructor to register\n   * @param {string} [name] - Optional name to register the constructor under, defaults to constructor.name\n   * @return {void}\n   * @throws {Error} If the constructor is not a function\n   */\n  register(constructor: ModelConstructor<M>, name?: string): void {\n    if (typeof constructor !== \"function\")\n      throw new Error(\n        \"Model registering failed. Missing Class name or constructor\"\n      );\n    name = name || constructor.name;\n    this.cache[name] = constructor;\n  }\n\n  /**\n   * @summary Gets a registered Model {@link ModelConstructor}\n   * @param {string} name\n   */\n  get(name: string): ModelConstructor<M> | undefined {\n    try {\n      return this.cache[name];\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: any) {\n      return undefined;\n    }\n  }\n\n  /**\n   * @param {Record<string, any>} obj\n   * @param {string} [clazz] when provided, it will attempt to find the matching constructor\n   *\n   * @throws Error If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property\n   */\n  build(obj: Record<string, any> = {}, clazz?: string): M {\n    if (!clazz && !this.testFunction(obj))\n      throw new Error(\"Provided obj is not a Model object\");\n    const name = clazz || Model.getMetadata(obj as any);\n    if (!(name in this.cache))\n      throw new Error(\n        `Provided class ${name} is not a registered Model object`\n      );\n    return new this.cache[name](obj);\n  }\n}\n\n/**\n * @summary Bulk Registers Models\n * @description Useful when using bundlers that might not evaluate all the code at once\n *\n * @template M extends Model\n * @param {Array<Constructor<M>> | Array<{name: string, constructor: Constructor<M>}>} [models]\n *\n * @memberOf module:decorator-validation\n * @category Model\n */\nexport function bulkModelRegister<M extends Model>(\n  ...models: (Constructor<M> | { name: string; constructor: Constructor<M> })[]\n) {\n  models.forEach(\n    (m: Constructor<M> | { name: string; constructor: Constructor<M> }) => {\n      const constructor: Constructor<M> = (\n        m.constructor ? m.constructor : m\n      ) as Constructor<M>;\n      Model.register(constructor, (m as Constructor<M>).name);\n    }\n  );\n}\n\n/**\n * @summary Abstract class representing a Validatable Model object\n * @description Meant to be used as a base class for all Model classes\n *\n * Model objects must:\n *  - Have all their required properties marked with '!';\n *  - Have all their optional properties marked as '?':\n *\n * @param {ModelArg<Model>} model base object from which to populate properties from\n *\n * @class Model\n * @category Model\n * @abstract\n * @implements Validatable\n * @implements Serializable\n *\n * @example\n *      class ClassName {\n *          @required()\n *          requiredPropertyName!: PropertyType;\n *\n *          optionalPropertyName?: PropertyType;\n *      }\n */\nexport abstract class Model\n  implements Validatable, Serializable, Hashable, Comparable<Model>\n{\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  protected constructor(arg?: ModelArg<Model>) {}\n\n  /**\n   * @description Validates the model object against its defined validation rules\n   * @summary Validates the object according to its decorated properties, returning any validation errors\n   *\n   * @param {any[]} [exceptions] - Properties in the object to be ignored for the validation. Marked as 'any' to allow for extension but expects strings\n   * @return {ModelErrorDefinition | undefined} - Returns a ModelErrorDefinition object if validation errors exist, otherwise undefined\n   */\n  public hasErrors(...exceptions: any[]): ModelErrorDefinition | undefined {\n    return validate(this, ...exceptions);\n  }\n\n  /**\n   * @description Determines if this model is equal to another object\n   * @summary Compare object equality recursively, checking all properties unless excluded\n   *\n   * @param {any} obj - Object to compare to\n   * @param {string[]} [exceptions] - Property names to be excluded from the comparison\n   * @return {boolean} - True if objects are equal, false otherwise\n   */\n  public equals(obj: any, ...exceptions: string[]): boolean {\n    return isEqual(this, obj, ...exceptions);\n  }\n\n  /**\n   * @description Converts the model to a serialized string representation\n   * @summary Returns the serialized model according to the currently defined {@link Serializer}\n   * \n   * @return {string} - The serialized string representation of the model\n   */\n  serialize(): string {\n    return Model.serialize(this);\n  }\n\n  /**\n   * @description Provides a human-readable string representation of the model\n   * @summary Override the implementation for js's 'toString()' to provide a more useful representation\n   * \n   * @return {string} - A string representation of the model including its class name and JSON representation\n   * @override\n   */\n  public toString(): string {\n    return this.constructor.name + \": \" + JSON.stringify(this, undefined, 2);\n  }\n\n  /**\n   * @description Generates a hash string for the model object\n   * @summary Defines a default implementation for object hash, relying on a basic implementation based on Java's string hash\n   * \n   * @return {string} - A hash string representing the model\n   */\n  public hash(): string {\n    return Model.hash(this);\n  }\n\n  /**\n   * @description Converts a serialized string back into a model instance\n   * @summary Deserializes a Model from its string representation\n   *\n   * @param {string} str - The serialized string to convert back to a model\n   * @return {any} - The deserialized model instance\n   * @throws {Error} If it fails to parse the string, or if it fails to build the model\n   */\n  static deserialize(str: string) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.SERIALIZATION),\n      this.constructor\n    );\n\n    if (metadata && metadata.serializer)\n      return Serialization.deserialize(\n        str,\n        metadata.serializer,\n        ...(metadata.args || [])\n      );\n    return Serialization.deserialize(str);\n  }\n\n  /**\n   * @description Copies properties from a source object to a model instance\n   * @summary Repopulates the Object properties with the ones from the new object\n   *\n   * @template T\n   * @param {T} self - The target model instance to update\n   * @param {T | Record<string, any>} [obj] - The source object containing properties to copy\n   * @return {T} - The updated model instance\n   */\n  static fromObject<T extends Model>(\n    self: T,\n    obj?: T | Record<string, any>\n  ): T {\n    if (!obj) obj = {};\n    for (const prop of Model.getAttributes(self)) {\n      (self as any)[prop] = (obj as any)[prop] || undefined;\n    }\n    return self;\n  }\n\n  /**\n   * @description Copies and rebuilds properties from a source object to a model instance, handling nested models\n   * @summary Repopulates the instance with properties from the new Model Object, recursively rebuilding nested models\n   *\n   * @template T\n   * @param {T} self - The target model instance to update\n   * @param {T | Record<string, any>} [obj] - The source object containing properties to copy\n   * @return {T} - The updated model instance with rebuilt nested models\n   * \n   * @mermaid\n   * sequenceDiagram\n   *   participant C as Client\n   *   participant M as Model.fromModel\n   *   participant B as Model.build\n   *   participant R as Reflection\n   *   \n   *   C->>M: fromModel(self, obj)\n   *   M->>M: Get attributes from self\n   *   loop For each property\n   *     M->>M: Copy property from obj to self\n   *     alt Property is a model\n   *       M->>M: Check if property is a model\n   *       M->>B: build(property, modelType)\n   *       B-->>M: Return built model\n   *     else Property is a complex type\n   *       M->>R: Get property decorators\n   *       R-->>M: Return decorators\n   *       M->>M: Filter type decorators\n   *       alt Property is Array/Set with list decorator\n   *         M->>M: Process each item in collection\n   *         loop For each item\n   *           M->>B: build(item, itemModelType)\n   *           B-->>M: Return built model\n   *         end\n   *       else Property is another model type\n   *         M->>B: build(property, propertyType)\n   *         B-->>M: Return built model\n   *       end\n   *     end\n   *   end\n   *   M-->>C: Return updated self\n   */\n  static fromModel<T extends Model>(self: T, obj?: T | Record<string, any>): T {\n    if (!obj) obj = {};\n\n    let decorators: DecoratorMetadata[], dec: DecoratorMetadata;\n\n    const props = Model.getAttributes(self);\n\n    for (const prop of props) {\n      (self as Record<string, any>)[prop] =\n        (obj as Record<string, any>)[prop] ?? undefined;\n      if (typeof (self as any)[prop] !== \"object\") continue;\n      const propM = Model.isPropertyModel(self, prop);\n      if (propM) {\n        try {\n          (self as Record<string, any>)[prop] = Model.build(\n            (self as Record<string, any>)[prop],\n            typeof propM === \"string\" ? propM : undefined\n          );\n        } catch (e: any) {\n          console.log(e);\n        }\n        continue;\n      }\n\n      const allDecorators: DecoratorMetadata[] =\n        Reflection.getPropertyDecorators(\n          ValidationKeys.REFLECT,\n          self,\n          prop\n        ).decorators;\n      decorators = allDecorators.filter(\n        (d: DecoratorMetadata) =>\n          [ModelKeys.TYPE, ValidationKeys.TYPE as string].indexOf(d.key) !== -1\n      );\n      if (!decorators || !decorators.length)\n        throw new Error(`failed to find decorators for property ${prop}`);\n      dec = decorators.pop() as DecoratorMetadata;\n      const clazz = dec.props.name\n        ? [dec.props.name]\n        : Array.isArray(dec.props.customTypes)\n          ? dec.props.customTypes\n          : [dec.props.customTypes];\n      const reserved = Object.values(ReservedModels).map((v) =>\n        v.toLowerCase()\n      ) as string[];\n\n      clazz.forEach((c) => {\n        if (reserved.indexOf(c.toLowerCase()) === -1)\n          try {\n            switch (c) {\n              case \"Array\":\n              case \"Set\":\n                if (allDecorators.length) {\n                  const listDec = allDecorators.find(\n                    (d) => d.key === ValidationKeys.LIST\n                  );\n                  if (listDec) {\n                    const clazzName = (listDec.props.clazz as string[]).find(\n                      (t: string) => !jsTypes.includes(t.toLowerCase())\n                    );\n                    if (c === \"Array\")\n                      (self as Record<string, any>)[prop] = (\n                        self as Record<string, any>\n                      )[prop].map((el: any) => {\n                        return [\"object\", \"function\"].includes(typeof el) &&\n                          clazzName\n                          ? Model.build(el, clazzName)\n                          : el;\n                      });\n                    if (c === \"Set\") {\n                      const s = new Set();\n                      for (const v of (self as Record<string, any>)[prop]) {\n                        if (\n                          [\"object\", \"function\"].includes(typeof v) &&\n                          clazzName\n                        ) {\n                          s.add(Model.build(v, clazzName));\n                        } else {\n                          s.add(v);\n                        }\n                      }\n                      (self as Record<string, any>)[prop] = s;\n                    }\n                  }\n                }\n                break;\n              default:\n                if ((self as Record<string, any>)[prop])\n                  (self as Record<string, any>)[prop] = Model.build(\n                    (self as any)[prop],\n                    c\n                  );\n            }\n          } catch (e: any) {\n            console.log(e);\n            // do nothing. we have no registry of this class\n          }\n      });\n    }\n    return self;\n  }\n\n  /**\n   * @description Configures the global model builder function\n   * @summary Sets the Global {@link ModelBuilderFunction} used for building model instances\n   *\n   * @param {ModelBuilderFunction} [builder] - The builder function to set as the global builder\n   * @return {void}\n   */\n  static setBuilder(builder?: ModelBuilderFunction) {\n    modelBuilderFunction = builder;\n  }\n\n  /**\n   * @description Retrieves the currently configured global model builder function\n   * @summary Returns the current global {@link ModelBuilderFunction} used for building model instances\n   *\n   * @return {ModelBuilderFunction | undefined} - The current global builder function or undefined if not set\n   */\n  static getBuilder(): ModelBuilderFunction | undefined {\n    return modelBuilderFunction;\n  }\n\n  /**\n   * @description Provides access to the current model registry\n   * @summary Returns the current {@link ModelRegistryManager} instance, creating one if it doesn't exist\n   *\n   * @return {ModelRegistry<any>} - The current model registry, defaults to a new {@link ModelRegistryManager} if not set\n   * @private\n   */\n  private static getRegistry() {\n    if (!actingModelRegistry) actingModelRegistry = new ModelRegistryManager();\n    return actingModelRegistry;\n  }\n\n  /**\n   * @description Configures the model registry to be used by the Model system\n   * @summary Sets the current model registry to a custom implementation\n   *\n   * @param {BuilderRegistry<any>} modelRegistry - The new implementation of Registry to use\n   * @return {void}\n   */\n  static setRegistry(modelRegistry: BuilderRegistry<any>) {\n    actingModelRegistry = modelRegistry;\n  }\n\n  /**\n   * @description Registers a model constructor with the model registry\n   * @summary Registers new model classes to make them available for serialization and deserialization\n   *\n   * @template T\n   * @param {ModelConstructor<T>} constructor - The model constructor to register\n   * @param {string} [name] - Optional name to register the constructor under, defaults to constructor.name\n   * @return {void}\n   * \n   * @see ModelRegistry\n   */\n  static register<T extends Model>(\n    constructor: ModelConstructor<T>,\n    name?: string\n  ): void {\n    return Model.getRegistry().register(constructor, name);\n  }\n\n  /**\n   * @description Retrieves a registered model constructor by name\n   * @summary Gets a registered Model {@link ModelConstructor} from the model registry\n   *\n   * @template T\n   * @param {string} name - The name of the model constructor to retrieve\n   * @return {ModelConstructor<T> | undefined} - The model constructor if found, undefined otherwise\n   * \n   * @see ModelRegistry\n   */\n  static get<T extends Model>(name: string): ModelConstructor<T> | undefined {\n    return Model.getRegistry().get(name);\n  }\n\n  /**\n   * @description Creates a model instance from a plain object\n   * @summary Builds a model instance using the model registry, optionally specifying the model class\n   *\n   * @template T\n   * @param {Record<string, any>} obj - The source object to build the model from\n   * @param {string} [clazz] - When provided, it will attempt to find the matching constructor by name\n   * @return {T} - The built model instance\n   * @throws {Error} If clazz is not found, or obj is not a {@link Model} meaning it has no {@link ModelKeys.ANCHOR} property\n   *\n   * @see ModelRegistry\n   */\n  static build<T extends Model>(\n    obj: Record<string, any> = {},\n    clazz?: string\n  ): T {\n    return Model.getRegistry().build(obj, clazz);\n  }\n\n  /**\n   * @description Retrieves the model metadata from a model instance\n   * @summary Gets the metadata associated with a model instance, typically the model class name\n   *\n   * @template M\n   * @param {M} model - The model instance to get metadata from\n   * @return {string} - The model metadata (typically the class name)\n   */\n  static getMetadata<M extends Model>(model: M) {\n    return getMetadata<M>(model);\n  }\n\n  /**\n   * @description Retrieves all attribute names from a model class or instance\n   * @summary Gets all attributes defined in a model, traversing the prototype chain to include inherited attributes\n   *\n   * @template V\n   * @param {Constructor<V> | V} model - The model class or instance to get attributes from\n   * @return {string[]} - Array of attribute names defined in the model\n   */\n  static getAttributes<V extends Model>(model: Constructor<V> | V) {\n    const result: string[] = [];\n    let prototype =\n      model instanceof Model\n        ? Object.getPrototypeOf(model)\n        : (model as any).prototype;\n    while (prototype != null) {\n      const props: string[] = prototype[ModelKeys.ATTRIBUTE];\n      if (props) {\n        result.push(...props);\n      }\n      prototype = Object.getPrototypeOf(prototype);\n    }\n    return result;\n  }\n\n  /**\n   * @description Compares two model instances for equality\n   * @summary Determines if two model instances are equal by comparing their properties\n   *\n   * @template M\n   * @param {M} obj1 - First model instance to compare\n   * @param {M} obj2 - Second model instance to compare\n   * @param {any[]} [exceptions] - Property names to exclude from comparison\n   * @return {boolean} - True if the models are equal, false otherwise\n   */\n  static equals<M extends Model>(obj1: M, obj2: M, ...exceptions: any[]) {\n    return isEqual(obj1, obj2, ...exceptions);\n  }\n\n  /**\n   * @description Validates a model instance against its validation rules\n   * @summary Checks if a model has validation errors, optionally ignoring specified properties\n   *\n   * @template M\n   * @param {M} model - The model instance to validate\n   * @param {string[]} [propsToIgnore] - Properties to exclude from validation\n   * @return {ModelErrorDefinition | undefined} - Returns validation errors if any, otherwise undefined\n   */\n  static hasErrors<M extends Model>(model: M, ...propsToIgnore: string[]) {\n    return validate(model, ...propsToIgnore);\n  }\n\n  /**\n   * @description Converts a model instance to a serialized string\n   * @summary Serializes a model instance using the configured serializer or the default one\n   *\n   * @template M\n   * @param {M} model - The model instance to serialize\n   * @return {string} - The serialized string representation of the model\n   */\n  static serialize<M extends Model>(model: M) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.SERIALIZATION),\n      model.constructor\n    );\n\n    if (metadata && metadata.serializer)\n      return Serialization.serialize(\n        this,\n        metadata.serializer,\n        ...(metadata.args || [])\n      );\n    return Serialization.serialize(model);\n  }\n\n  /**\n   * @description Generates a hash string for a model instance\n   * @summary Creates a hash representation of a model using the configured algorithm or the default one\n   *\n   * @template M\n   * @param {M} model - The model instance to hash\n   * @return {string} - The hash string representing the model\n   */\n  static hash<M extends Model>(model: M) {\n    const metadata = Reflect.getMetadata(\n      Model.key(ModelKeys.HASHING),\n      model.constructor\n    );\n\n    if (metadata && metadata.algorithm)\n      return Hashing.hash(model, metadata.algorithm, ...(metadata.args || []));\n    return Hashing.hash(model);\n  }\n  /**\n   * @description Creates a metadata key for use with the Reflection API\n   * @summary Builds the key to store as Metadata under Reflections\n   *\n   * @param {string} str - The base key to concatenate with the model reflection prefix\n   * @return {string} - The complete metadata key\n   */\n  static key(str: string) {\n    return getModelKey(str);\n  }\n\n  /**\n   * @description Determines if an object is a model instance or has model metadata\n   * @summary Checks whether a given object is either an instance of the Model class or\n   * has model metadata attached to it. This function is essential for serialization and\n   * deserialization processes, as it helps identify model objects that need special handling.\n   * It safely handles potential errors during metadata retrieval.\n   *\n   * @param {Record<string, any>} target - The object to check\n   * @return {boolean} True if the object is a model instance or has model metadata, false otherwise\n   *\n   * @example\n   * ```typescript\n   * // Check if an object is a model\n   * const user = new User({ name: \"John\" });\n   * const isUserModel = isModel(user); // true\n   *\n   * // Check a plain object\n   * const plainObject = { name: \"John\" };\n   * const isPlainObjectModel = isModel(plainObject); // false\n   * ```\n   */\n  static isModel(target: Record<string, any>) {\n    try {\n      return target instanceof Model || !!Model.getMetadata(target as any);\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e: any) {\n      return false;\n    }\n  }\n\n  /**\n   * @description Checks if a property of a model is itself a model or has a model type\n   * @summary Determines whether a specific property of a model instance is either a model instance\n   * or has a type that is registered as a model\n   *\n   * @template M\n   * @param {M} target - The model instance to check\n   * @param {string} attribute - The property name to check\n   * @return {boolean | string | undefined} - Returns true if the property is a model instance,\n   * the model name if the property has a model type, or undefined if not a model\n   */\n  static isPropertyModel<M extends Model>(\n    target: M,\n    attribute: string\n  ): boolean | string | undefined {\n    if (Model.isModel((target as Record<string, any>)[attribute])) return true;\n    const metadata = Reflect.getMetadata(ModelKeys.TYPE, target, attribute);\n    return Model.get(metadata.name) ? metadata.name : undefined;\n  }\n}\n"]}
@@ -2,14 +2,16 @@ import { Model } from "./Model";
2
2
  /**
3
3
  * @summary Helper Function to override constructors
4
4
  *
5
+ * @template M the model instance type
6
+ *
5
7
  * @param {Function} constructor
6
8
  * @param {any[]} [args]
7
- * @return {T} the new instance
9
+ * @return {M} the new instance
8
10
  *
9
11
  * @function construct
10
12
  * @memberOf module:decorator-validation
11
13
  */
12
- export declare function construct<T extends Model>(constructor: any, ...args: any[]): T;
14
+ export declare function construct<M extends Model>(constructor: any, ...args: any[]): M;
13
15
  /**
14
16
  * @summary Recursively finds the last prototype before Object
15
17
  * @param {object} obj
@@ -2,9 +2,11 @@ import { Model } from "./Model";
2
2
  /**
3
3
  * @summary Helper Function to override constructors
4
4
  *
5
+ * @template M the model instance type
6
+ *
5
7
  * @param {Function} constructor
6
8
  * @param {any[]} [args]
7
- * @return {T} the new instance
9
+ * @return {M} the new instance
8
10
  *
9
11
  * @function construct
10
12
  * @memberOf module:decorator-validation
@@ -61,4 +63,4 @@ export function bindModelPrototype(obj) {
61
63
  }
62
64
  throw new Error("Could not find proper prototype to bind");
63
65
  }
64
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RydWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL21vZGVsL2NvbnN0cnVjdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRWhDOzs7Ozs7Ozs7R0FTRztBQUNILE1BQU0sVUFBVSxTQUFTLENBQ3ZCLFdBQWdCLEVBQ2hCLEdBQUcsSUFBVztJQUVkLE1BQU0sT0FBTyxHQUFHLENBQUMsR0FBRyxJQUFXLEVBQUUsRUFBRSxDQUFDLElBQUksV0FBVyxDQUFDLEdBQUcsSUFBSSxDQUFDLENBQUM7SUFDN0QsT0FBTyxDQUFDLFNBQVMsR0FBRyxXQUFXLENBQUMsU0FBUyxDQUFDO0lBQzFDLE9BQU8sT0FBTyxDQUFDLEdBQUcsSUFBSSxDQUFDLENBQUM7QUFDMUIsQ0FBQztBQUVEOzs7Ozs7R0FNRztBQUNILE1BQU0sVUFBVSx5QkFBeUIsQ0FBQyxHQUFXO0lBQ25ELElBQUksU0FBUyxHQUFRLE1BQU0sQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDaEQsSUFBSSxTQUFTLEtBQUssTUFBTSxDQUFDLFNBQVM7UUFBRSxPQUFPLEdBQUcsQ0FBQztJQUMvQyxPQUFPLFNBQVMsS0FBSyxNQUFNLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDdEMsU0FBUyxHQUFHLE1BQU0sQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDN0MsSUFBSSxTQUFTLEtBQUssTUFBTSxDQUFDLFNBQVM7WUFBRSxPQUFPLFNBQVMsQ0FBQztRQUNyRCxJQUFJLE1BQU0sQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLEtBQUssTUFBTSxDQUFDLFNBQVM7WUFBRSxPQUFPLFNBQVMsQ0FBQztJQUM5RSxDQUFDO0lBQ0QsTUFBTSxJQUFJLEtBQUssQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO0FBQ3JELENBQUM7QUFFRDs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxVQUFVLGtCQUFrQixDQUFDLEdBQVk7SUFDN0MsSUFBSSxHQUFHLFlBQVksS0FBSztRQUFFLE9BQU87SUFFakMsU0FBUyxhQUFhLENBQUMsYUFBc0IsRUFBRSxTQUFpQjtRQUM5RCxNQUFNLENBQUMsY0FBYyxDQUFDLGFBQWEsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQsTUFBTSxTQUFTLEdBQVEsTUFBTSxDQUFDLGNBQWMsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNsRCxJQUFJLFNBQVMsS0FBSyxNQUFNLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDbkMsT0FBTyxhQUFhLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUM3QyxDQUFDO0lBQ0QsT0FBTyxTQUFTLEtBQUssTUFBTSxDQUFDLFNBQVMsRUFBRSxDQUFDO1FBQ3RDLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxjQUFjLENBQUMsU0FBUyxDQUFDLENBQUM7UUFDOUMsSUFDRSxJQUFJLEtBQUssTUFBTSxDQUFDLFNBQVM7WUFDekIsTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsS0FBSyxNQUFNLENBQUMsU0FBUyxFQUNoRCxDQUFDO1lBQ0QsT0FBTyxhQUFhLENBQUMsU0FBUyxFQUFFLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUNuRCxDQUFDO0lBQ0gsQ0FBQztJQUNELE1BQU0sSUFBSSxLQUFLLENBQUMseUNBQXlDLENBQUMsQ0FBQztBQUM3RCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiLi9Nb2RlbFwiO1xuXG4vKipcbiAqIEBzdW1tYXJ5IEhlbHBlciBGdW5jdGlvbiB0byBvdmVycmlkZSBjb25zdHJ1Y3RvcnNcbiAqXG4gKiBAcGFyYW0ge0Z1bmN0aW9ufSBjb25zdHJ1Y3RvclxuICogQHBhcmFtIHthbnlbXX0gW2FyZ3NdXG4gKiBAcmV0dXJuIHtUfSB0aGUgbmV3IGluc3RhbmNlXG4gKlxuICogQGZ1bmN0aW9uIGNvbnN0cnVjdFxuICogQG1lbWJlck9mIG1vZHVsZTpkZWNvcmF0b3ItdmFsaWRhdGlvblxuICovXG5leHBvcnQgZnVuY3Rpb24gY29uc3RydWN0PFQgZXh0ZW5kcyBNb2RlbD4oXG4gIGNvbnN0cnVjdG9yOiBhbnksXG4gIC4uLmFyZ3M6IGFueVtdXG4pOiBUIHtcbiAgY29uc3QgX2NvbnN0ciA9ICguLi5hcmd6OiBhbnlbXSkgPT4gbmV3IGNvbnN0cnVjdG9yKC4uLmFyZ3opO1xuICBfY29uc3RyLnByb3RvdHlwZSA9IGNvbnN0cnVjdG9yLnByb3RvdHlwZTtcbiAgcmV0dXJuIF9jb25zdHIoLi4uYXJncyk7XG59XG5cbi8qKlxuICogQHN1bW1hcnkgUmVjdXJzaXZlbHkgZmluZHMgdGhlIGxhc3QgcHJvdG90eXBlIGJlZm9yZSBPYmplY3RcbiAqIEBwYXJhbSB7b2JqZWN0fSBvYmpcbiAqXG4gKiBAZnVuY3Rpb24gZmluZExhc3RQcm90b0JlZm9yZU9iamVjdFxuICogQG1lbWJlck9mIG1vZHVsZTpkZWNvcmF0b3ItdmFsaWRhdGlvblxuICovXG5leHBvcnQgZnVuY3Rpb24gZmluZExhc3RQcm90b0JlZm9yZU9iamVjdChvYmo6IG9iamVjdCk6IG9iamVjdCB7XG4gIGxldCBwcm90b3R5cGU6IGFueSA9IE9iamVjdC5nZXRQcm90b3R5cGVPZihvYmopO1xuICBpZiAocHJvdG90eXBlID09PSBPYmplY3QucHJvdG90eXBlKSByZXR1cm4gb2JqO1xuICB3aGlsZSAocHJvdG90eXBlICE9PSBPYmplY3QucHJvdG90eXBlKSB7XG4gICAgcHJvdG90eXBlID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKHByb3RvdHlwZSk7XG4gICAgaWYgKHByb3RvdHlwZSA9PT0gT2JqZWN0LnByb3RvdHlwZSkgcmV0dXJuIHByb3RvdHlwZTtcbiAgICBpZiAoT2JqZWN0LmdldFByb3RvdHlwZU9mKHByb3RvdHlwZSkgPT09IE9iamVjdC5wcm90b3R5cGUpIHJldHVybiBwcm90b3R5cGU7XG4gIH1cbiAgdGhyb3cgbmV3IEVycm9yKFwiQ291bGQgbm90IGZpbmQgcHJvcGVyIHByb3RvdHlwZVwiKTtcbn1cblxuLyoqXG4gKiBAc3VtYXJ5IGJpbmRzIHRoZSB7QGxpbmsgTW9kZWx9IGNsYXNzIGFzIGEgcm9vdCBwcm90b3R5cGUgb2YgdGhlIHByb3ZpZGVkIGluc3RhbmNlXG4gKlxuICogQHBhcmFtIHt1bmtub3dufSBvYmpcbiAqXG4gKiBAZnVuY3Rpb24gYmluZE1vZGVsUHJvdG90eXBlXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmRlY29yYXRvci12YWxpZGF0aW9uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBiaW5kTW9kZWxQcm90b3R5cGUob2JqOiB1bmtub3duKSB7XG4gIGlmIChvYmogaW5zdGFuY2VvZiBNb2RlbCkgcmV0dXJuO1xuXG4gIGZ1bmN0aW9uIGJpbmRQcm90b3R5cGUob2JqVG9PdmVycmlkZTogdW5rbm93biwgcHJvdG90eXBlOiBvYmplY3QpIHtcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2Yob2JqVG9PdmVycmlkZSwgcHJvdG90eXBlKTtcbiAgfVxuXG4gIGNvbnN0IHByb3RvdHlwZTogYW55ID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKG9iaik7XG4gIGlmIChwcm90b3R5cGUgPT09IE9iamVjdC5wcm90b3R5cGUpIHtcbiAgICByZXR1cm4gYmluZFByb3RvdHlwZShvYmosIE1vZGVsLnByb3RvdHlwZSk7XG4gIH1cbiAgd2hpbGUgKHByb3RvdHlwZSAhPT0gT2JqZWN0LnByb3RvdHlwZSkge1xuICAgIGNvbnN0IHByb3QgPSBPYmplY3QuZ2V0UHJvdG90eXBlT2YocHJvdG90eXBlKTtcbiAgICBpZiAoXG4gICAgICBwcm90ID09PSBPYmplY3QucHJvdG90eXBlIHx8XG4gICAgICBPYmplY3QuZ2V0UHJvdG90eXBlT2YocHJvdCkgPT09IE9iamVjdC5wcm90b3R5cGVcbiAgICApIHtcbiAgICAgIHJldHVybiBiaW5kUHJvdG90eXBlKHByb3RvdHlwZSwgTW9kZWwucHJvdG90eXBlKTtcbiAgICB9XG4gIH1cbiAgdGhyb3cgbmV3IEVycm9yKFwiQ291bGQgbm90IGZpbmQgcHJvcGVyIHByb3RvdHlwZSB0byBiaW5kXCIpO1xufVxuIl19
66
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RydWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL21vZGVsL2NvbnN0cnVjdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sU0FBUyxDQUFDO0FBRWhDOzs7Ozs7Ozs7OztHQVdHO0FBQ0gsTUFBTSxVQUFVLFNBQVMsQ0FDdkIsV0FBZ0IsRUFDaEIsR0FBRyxJQUFXO0lBRWQsTUFBTSxPQUFPLEdBQUcsQ0FBQyxHQUFHLElBQVcsRUFBRSxFQUFFLENBQUMsSUFBSSxXQUFXLENBQUMsR0FBRyxJQUFJLENBQUMsQ0FBQztJQUM3RCxPQUFPLENBQUMsU0FBUyxHQUFHLFdBQVcsQ0FBQyxTQUFTLENBQUM7SUFDMUMsT0FBTyxPQUFPLENBQUMsR0FBRyxJQUFJLENBQUMsQ0FBQztBQUMxQixDQUFDO0FBRUQ7Ozs7OztHQU1HO0FBQ0gsTUFBTSxVQUFVLHlCQUF5QixDQUFDLEdBQVc7SUFDbkQsSUFBSSxTQUFTLEdBQVEsTUFBTSxDQUFDLGNBQWMsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNoRCxJQUFJLFNBQVMsS0FBSyxNQUFNLENBQUMsU0FBUztRQUFFLE9BQU8sR0FBRyxDQUFDO0lBQy9DLE9BQU8sU0FBUyxLQUFLLE1BQU0sQ0FBQyxTQUFTLEVBQUUsQ0FBQztRQUN0QyxTQUFTLEdBQUcsTUFBTSxDQUFDLGNBQWMsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUM3QyxJQUFJLFNBQVMsS0FBSyxNQUFNLENBQUMsU0FBUztZQUFFLE9BQU8sU0FBUyxDQUFDO1FBQ3JELElBQUksTUFBTSxDQUFDLGNBQWMsQ0FBQyxTQUFTLENBQUMsS0FBSyxNQUFNLENBQUMsU0FBUztZQUFFLE9BQU8sU0FBUyxDQUFDO0lBQzlFLENBQUM7SUFDRCxNQUFNLElBQUksS0FBSyxDQUFDLGlDQUFpQyxDQUFDLENBQUM7QUFDckQsQ0FBQztBQUVEOzs7Ozs7O0dBT0c7QUFDSCxNQUFNLFVBQVUsa0JBQWtCLENBQUMsR0FBWTtJQUM3QyxJQUFJLEdBQUcsWUFBWSxLQUFLO1FBQUUsT0FBTztJQUVqQyxTQUFTLGFBQWEsQ0FBQyxhQUFzQixFQUFFLFNBQWlCO1FBQzlELE1BQU0sQ0FBQyxjQUFjLENBQUMsYUFBYSxFQUFFLFNBQVMsQ0FBQyxDQUFDO0lBQ2xELENBQUM7SUFFRCxNQUFNLFNBQVMsR0FBUSxNQUFNLENBQUMsY0FBYyxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ2xELElBQUksU0FBUyxLQUFLLE1BQU0sQ0FBQyxTQUFTLEVBQUUsQ0FBQztRQUNuQyxPQUFPLGFBQWEsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFDRCxPQUFPLFNBQVMsS0FBSyxNQUFNLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDdEMsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLGNBQWMsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUM5QyxJQUNFLElBQUksS0FBSyxNQUFNLENBQUMsU0FBUztZQUN6QixNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxLQUFLLE1BQU0sQ0FBQyxTQUFTLEVBQ2hELENBQUM7WUFDRCxPQUFPLGFBQWEsQ0FBQyxTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQ25ELENBQUM7SUFDSCxDQUFDO0lBQ0QsTUFBTSxJQUFJLEtBQUssQ0FBQyx5Q0FBeUMsQ0FBQyxDQUFDO0FBQzdELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNb2RlbCB9IGZyb20gXCIuL01vZGVsXCI7XG5cbi8qKlxuICogQHN1bW1hcnkgSGVscGVyIEZ1bmN0aW9uIHRvIG92ZXJyaWRlIGNvbnN0cnVjdG9yc1xuICpcbiAqIEB0ZW1wbGF0ZSBNIHRoZSBtb2RlbCBpbnN0YW5jZSB0eXBlXG4gKlxuICogQHBhcmFtIHtGdW5jdGlvbn0gY29uc3RydWN0b3JcbiAqIEBwYXJhbSB7YW55W119IFthcmdzXVxuICogQHJldHVybiB7TX0gdGhlIG5ldyBpbnN0YW5jZVxuICpcbiAqIEBmdW5jdGlvbiBjb25zdHJ1Y3RcbiAqIEBtZW1iZXJPZiBtb2R1bGU6ZGVjb3JhdG9yLXZhbGlkYXRpb25cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNvbnN0cnVjdDxNIGV4dGVuZHMgTW9kZWw+KFxuICBjb25zdHJ1Y3RvcjogYW55LFxuICAuLi5hcmdzOiBhbnlbXVxuKTogTSB7XG4gIGNvbnN0IF9jb25zdHIgPSAoLi4uYXJnejogYW55W10pID0+IG5ldyBjb25zdHJ1Y3RvciguLi5hcmd6KTtcbiAgX2NvbnN0ci5wcm90b3R5cGUgPSBjb25zdHJ1Y3Rvci5wcm90b3R5cGU7XG4gIHJldHVybiBfY29uc3RyKC4uLmFyZ3MpO1xufVxuXG4vKipcbiAqIEBzdW1tYXJ5IFJlY3Vyc2l2ZWx5IGZpbmRzIHRoZSBsYXN0IHByb3RvdHlwZSBiZWZvcmUgT2JqZWN0XG4gKiBAcGFyYW0ge29iamVjdH0gb2JqXG4gKlxuICogQGZ1bmN0aW9uIGZpbmRMYXN0UHJvdG9CZWZvcmVPYmplY3RcbiAqIEBtZW1iZXJPZiBtb2R1bGU6ZGVjb3JhdG9yLXZhbGlkYXRpb25cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpbmRMYXN0UHJvdG9CZWZvcmVPYmplY3Qob2JqOiBvYmplY3QpOiBvYmplY3Qge1xuICBsZXQgcHJvdG90eXBlOiBhbnkgPSBPYmplY3QuZ2V0UHJvdG90eXBlT2Yob2JqKTtcbiAgaWYgKHByb3RvdHlwZSA9PT0gT2JqZWN0LnByb3RvdHlwZSkgcmV0dXJuIG9iajtcbiAgd2hpbGUgKHByb3RvdHlwZSAhPT0gT2JqZWN0LnByb3RvdHlwZSkge1xuICAgIHByb3RvdHlwZSA9IE9iamVjdC5nZXRQcm90b3R5cGVPZihwcm90b3R5cGUpO1xuICAgIGlmIChwcm90b3R5cGUgPT09IE9iamVjdC5wcm90b3R5cGUpIHJldHVybiBwcm90b3R5cGU7XG4gICAgaWYgKE9iamVjdC5nZXRQcm90b3R5cGVPZihwcm90b3R5cGUpID09PSBPYmplY3QucHJvdG90eXBlKSByZXR1cm4gcHJvdG90eXBlO1xuICB9XG4gIHRocm93IG5ldyBFcnJvcihcIkNvdWxkIG5vdCBmaW5kIHByb3BlciBwcm90b3R5cGVcIik7XG59XG5cbi8qKlxuICogQHN1bWFyeSBiaW5kcyB0aGUge0BsaW5rIE1vZGVsfSBjbGFzcyBhcyBhIHJvb3QgcHJvdG90eXBlIG9mIHRoZSBwcm92aWRlZCBpbnN0YW5jZVxuICpcbiAqIEBwYXJhbSB7dW5rbm93bn0gb2JqXG4gKlxuICogQGZ1bmN0aW9uIGJpbmRNb2RlbFByb3RvdHlwZVxuICogQG1lbWJlck9mIG1vZHVsZTpkZWNvcmF0b3ItdmFsaWRhdGlvblxuICovXG5leHBvcnQgZnVuY3Rpb24gYmluZE1vZGVsUHJvdG90eXBlKG9iajogdW5rbm93bikge1xuICBpZiAob2JqIGluc3RhbmNlb2YgTW9kZWwpIHJldHVybjtcblxuICBmdW5jdGlvbiBiaW5kUHJvdG90eXBlKG9ialRvT3ZlcnJpZGU6IHVua25vd24sIHByb3RvdHlwZTogb2JqZWN0KSB7XG4gICAgT2JqZWN0LnNldFByb3RvdHlwZU9mKG9ialRvT3ZlcnJpZGUsIHByb3RvdHlwZSk7XG4gIH1cblxuICBjb25zdCBwcm90b3R5cGU6IGFueSA9IE9iamVjdC5nZXRQcm90b3R5cGVPZihvYmopO1xuICBpZiAocHJvdG90eXBlID09PSBPYmplY3QucHJvdG90eXBlKSB7XG4gICAgcmV0dXJuIGJpbmRQcm90b3R5cGUob2JqLCBNb2RlbC5wcm90b3R5cGUpO1xuICB9XG4gIHdoaWxlIChwcm90b3R5cGUgIT09IE9iamVjdC5wcm90b3R5cGUpIHtcbiAgICBjb25zdCBwcm90ID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKHByb3RvdHlwZSk7XG4gICAgaWYgKFxuICAgICAgcHJvdCA9PT0gT2JqZWN0LnByb3RvdHlwZSB8fFxuICAgICAgT2JqZWN0LmdldFByb3RvdHlwZU9mKHByb3QpID09PSBPYmplY3QucHJvdG90eXBlXG4gICAgKSB7XG4gICAgICByZXR1cm4gYmluZFByb3RvdHlwZShwcm90b3R5cGUsIE1vZGVsLnByb3RvdHlwZSk7XG4gICAgfVxuICB9XG4gIHRocm93IG5ldyBFcnJvcihcIkNvdWxkIG5vdCBmaW5kIHByb3BlciBwcm90b3R5cGUgdG8gYmluZFwiKTtcbn1cbiJdfQ==