@decaf-ts/injectable-decorators 1.6.1 → 1.6.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.
@@ -1,70 +1,98 @@
1
- import { InjectablesKeys } from "./constants";
2
- import { Injectables } from "./Injectables";
3
- import { getTypeFromDecorator } from "./utils";
1
+ import { InjectablesKeys } from "./constants.js";
2
+ import { Injectables } from "./Injectables.js";
3
+ import { getTypeFromDecorator } from "./utils.js";
4
4
  /**
5
- * @summary Return the reflection key for injectables
6
- *
7
- * @param {string} key
5
+ * @description Generates a fully qualified reflection metadata key.
6
+ * @summary Returns the reflection key for injectables by prefixing the provided key with the base reflection key.
7
+ * @param {string} key The key to be prefixed
8
+ * @return {string} The fully qualified reflection key
8
9
  * @function getInjectKey
9
- *
10
10
  * @memberOf module:injectable-decorators
11
11
  */
12
12
  const getInjectKey = (key) => InjectablesKeys.REFLECT + key;
13
13
  /**
14
- * @summary Defines a class as an injectable
14
+ * @description Decorator that marks a class as available for dependency injection.
15
+ * @summary Defines a class as an injectable singleton that can be retrieved from the registry.
16
+ * When applied to a class, replaces its constructor with one that returns a singleton instance.
15
17
  *
16
- * @param {string} [category] defaults to the class Name. (Useful when minification occours and names are changed so we can no longer rely on the class name, or when we want to upcast the Object)
17
- * @param {boolean} [force] defines if the injectable should override the already existing instance (if any). (only meant for extending decorators
18
- * @param instanceCallback
18
+ * @param {string} [category] Defaults to the class name. Useful when minification occurs and names are changed,
19
+ * or when you want to upcast the object to a different type.
20
+ * @param {boolean} [force] Defines if the injectable should override an already existing instance (if any).
21
+ * Only meant for extending decorators.
22
+ * @param {Function} [instanceCallback] Optional callback function that will be called with the instance after creation.
23
+ * @return {Function} A decorator function that transforms the class into an injectable.
19
24
  *
20
25
  * @function injectable
26
+ * @category Class Decorators
27
+ *
28
+ * @mermaid
29
+ * sequenceDiagram
30
+ * participant Client
31
+ * participant Decorator
32
+ * participant Injectables
33
+ *
34
+ * Client->>Decorator: @injectable()
35
+ * Decorator->>Decorator: Create new constructor
21
36
  *
22
- * @memberOf module:injectable-decorators.Decorators
37
+ * Note over Decorator: When new instance requested
38
+ * Decorator->>Injectables: get(name)
39
+ * alt Instance exists
40
+ * Injectables-->>Decorator: Return existing instance
41
+ * else No instance
42
+ * Decorator->>Injectables: register(original, name)
43
+ * Decorator->>Injectables: get(name)
44
+ * Injectables-->>Decorator: Return new instance
45
+ * opt Has callback
46
+ * Decorator->>Decorator: Call instanceCallback
47
+ * end
48
+ * end
49
+ * Decorator->>Decorator: Define metadata
50
+ * Decorator-->>Client: Return instance
23
51
  */
24
- export const injectable = (category = undefined, force = false, instanceCallback) =>
25
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
26
- (original, propertyKey) => {
27
- const name = category || original.name;
28
- // the new constructor behaviour
29
- const newConstructor = function (...args) {
30
- let inj = Injectables.get(name, ...args);
31
- if (!inj) {
32
- Injectables.register(original, name, true, force);
33
- inj = Injectables.get(name, ...args);
34
- if (!inj)
35
- return undefined;
36
- if (instanceCallback)
37
- try {
38
- instanceCallback(inj);
39
- }
40
- catch (e) {
41
- console.error(`Failed to call injectable callback for ${name}: ${e}`);
42
- }
43
- }
44
- const metadata = Object.assign({}, {
45
- class: name,
52
+ export function injectable(category = undefined, force = false, instanceCallback) {
53
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
54
+ return (original, propertyKey) => {
55
+ const name = category || original.name;
56
+ // the new constructor behaviour
57
+ const newConstructor = function (...args) {
58
+ let inj = Injectables.get(name, ...args);
59
+ if (!inj) {
60
+ Injectables.register(original, name, true, force);
61
+ inj = Injectables.get(name, ...args);
62
+ if (!inj)
63
+ return undefined;
64
+ if (instanceCallback)
65
+ try {
66
+ instanceCallback(inj);
67
+ }
68
+ catch (e) {
69
+ console.error(`Failed to call injectable callback for ${name}: ${e}`);
70
+ }
71
+ }
72
+ const metadata = Object.assign({}, {
73
+ class: name,
74
+ });
75
+ Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECTABLE), metadata, inj.constructor);
76
+ return inj;
77
+ };
78
+ // copy prototype so instanceof operator still works
79
+ newConstructor.prototype = original.prototype;
80
+ // newConstructor.__proto__ = original.__proto__;
81
+ // Sets the proper constructor name for type verification
82
+ Object.defineProperty(newConstructor, "name", {
83
+ writable: false,
84
+ enumerable: true,
85
+ configurable: false,
86
+ value: original.prototype.constructor.name,
46
87
  });
47
- Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECTABLE), metadata, inj.constructor);
48
- return inj;
88
+ // return new constructor (will override original)
89
+ return newConstructor;
49
90
  };
50
- // copy prototype so instanceof operator still works
51
- newConstructor.prototype = original.prototype;
52
- // newConstructor.__proto__ = original.__proto__;
53
- // Sets the proper constructor name for type verification
54
- Object.defineProperty(newConstructor, "name", {
55
- writable: false,
56
- enumerable: true,
57
- configurable: false,
58
- value: original.prototype.constructor.name,
59
- });
60
- // return new constructor (will override original)
61
- return newConstructor;
62
- };
91
+ }
63
92
  /**
64
- * @summary Allows for the injection of an {@link injectable} decorated dependency
65
- * @description the property must be typed for the requested dependency.
66
- *
67
- * Only concrete classes. No generics are supported
93
+ * @description Property decorator that injects a dependency into a class property.
94
+ * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
95
+ * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.
68
96
  *
69
97
  * Injected properties should be described like so:
70
98
  * <pre>
@@ -79,53 +107,86 @@ export const injectable = (category = undefined, force = false, instanceCallback
79
107
  * </pre>
80
108
  *
81
109
  * where InjectableClass is the class you want to inject.
82
- * Notice the use of '!:' to ensure the transpiler the property will be set outside the constructor but will always be defined
83
- * For project where minification occours, you should use the category param to ensure the name is the same throughout
110
+ * Notice the use of '!:' to ensure the transpiler the property will be set outside the constructor but will always be defined.
111
+ * For projects where minification occurs, you should use the category param to ensure the name is the same throughout.
84
112
  *
85
- * @param {string} [category] defaults to the class Name. (Useful when minification occours and names are changed so we can no longer rely on the class name, or when we want to upcast the Object)
86
- * @param {InstanceTransformer} [transformer]
113
+ * @param {string} [category] Defaults to the class name derived from the property type. Useful when minification occurs
114
+ * and names are changed, or when you want to upcast the object to a different type.
115
+ * @param {InstanceTransformer} [transformer] Optional function to transform the injectable instance before it's injected.
116
+ * @return {Function} A property decorator function that sets up the dependency injection.
87
117
  *
88
118
  * @function inject
119
+ * @category Property Decorators
120
+ *
121
+ * @mermaid
122
+ * sequenceDiagram
123
+ * participant Client
124
+ * participant Decorator
125
+ * participant Injectables
89
126
  *
90
- * @memberOf module:injectable-decorators.Decorators
127
+ * Client->>Decorator: @inject()
128
+ * Decorator->>Decorator: Get type from property
129
+ * Decorator->>Decorator: Define metadata
130
+ * Decorator->>Decorator: Define property getter
131
+ *
132
+ * Note over Decorator: When property accessed
133
+ * Client->>Decorator: access property
134
+ * Decorator->>Decorator: Check if instance exists
135
+ * alt Instance exists in WeakMap
136
+ * Decorator-->>Client: Return cached instance
137
+ * else No instance
138
+ * Decorator->>Injectables: get(name)
139
+ * alt Injectable found
140
+ * Injectables-->>Decorator: Return injectable instance
141
+ * opt Has transformer
142
+ * Decorator->>Decorator: Call transformer
143
+ * end
144
+ * Decorator->>Decorator: Store in WeakMap
145
+ * Decorator-->>Client: Return instance
146
+ * else No injectable
147
+ * Decorator-->>Client: Throw error
148
+ * end
149
+ * end
91
150
  */
92
- export const inject = (category, transformer) => (target, propertyKey) => {
93
- const values = new WeakMap();
94
- const name = category || getTypeFromDecorator(target, propertyKey);
95
- if (!name)
96
- throw new Error(`Could not get Type from decorator`);
97
- Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECT), {
98
- injectable: name,
99
- }, target, propertyKey);
100
- Object.defineProperty(target, propertyKey, {
101
- configurable: true,
102
- get() {
103
- const descriptor = Object.getOwnPropertyDescriptor(target, propertyKey);
104
- if (descriptor.configurable) {
105
- Object.defineProperty(this, propertyKey, {
106
- enumerable: true,
107
- configurable: false,
108
- get() {
109
- let obj = values.get(this);
110
- if (!obj) {
111
- obj = Injectables.get(name);
112
- if (!obj)
113
- throw new Error(`Could not get Injectable ${name} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`);
114
- if (transformer)
115
- try {
116
- obj = transformer(obj, target);
117
- }
118
- catch (e) {
119
- console.error(e);
120
- }
121
- values.set(this, obj);
122
- }
123
- return obj;
124
- },
125
- });
126
- return this[propertyKey];
127
- }
128
- },
129
- });
130
- };
131
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import { InjectablesKeys } from \"./constants\";\nimport { Injectables } from \"./Injectables\";\nimport { getTypeFromDecorator } from \"./utils\";\n\n/**\n * @summary Return the reflection key for injectables\n *\n * @param {string} key\n * @function getInjectKey\n *\n * @memberOf module:injectable-decorators\n */\nconst getInjectKey = (key: string) => InjectablesKeys.REFLECT + key;\n\n/**\n * @summary Defines a class as an injectable\n *\n * @param {string} [category] defaults to the class Name. (Useful when minification occours and names are changed so we can no longer rely on the class name, or when we want to upcast the Object)\n * @param {boolean} [force] defines if the injectable should override the already existing instance (if any). (only meant for extending decorators\n * @param instanceCallback\n *\n * @function injectable\n *\n * @memberOf module:injectable-decorators.Decorators\n */\nexport const injectable =\n  (\n    category: string | undefined = undefined,\n    force: boolean = false,\n    instanceCallback?: (instance: any, ...args: any[]) => void\n  ) =>\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  (original: any, propertyKey?: any) => {\n    const name = category || original.name;\n    // the new constructor behaviour\n    const newConstructor: any = function (...args: any[]) {\n      let inj: any = Injectables.get<any>(name, ...args);\n      if (!inj) {\n        Injectables.register(original, name, true, force);\n        inj = Injectables.get<any>(name, ...args);\n        if (!inj) return undefined;\n\n        if (instanceCallback)\n          try {\n            instanceCallback(inj);\n          } catch (e: any) {\n            console.error(\n              `Failed to call injectable callback for ${name}: ${e}`\n            );\n          }\n      }\n\n      const metadata = Object.assign(\n        {},\n        {\n          class: name,\n        }\n      );\n\n      Reflect.defineMetadata(\n        getInjectKey(InjectablesKeys.INJECTABLE),\n        metadata,\n        inj.constructor\n      );\n\n      return inj;\n    };\n\n    // copy prototype so instanceof operator still works\n    newConstructor.prototype = original.prototype;\n    // newConstructor.__proto__ = original.__proto__;\n    // Sets the proper constructor name for type verification\n    Object.defineProperty(newConstructor, \"name\", {\n      writable: false,\n      enumerable: true,\n      configurable: false,\n      value: original.prototype.constructor.name,\n    });\n    // return new constructor (will override original)\n    return newConstructor;\n  };\n\n/**\n * @summary function witch transforms a cached {@link injectable}\n *\n * @param {any} injectable\n * @param {any} obj the obj the injectable will be injected on\n *\n * @memberOf module:injectable-decorators\n */\nexport type InstanceTransformer = (injectable: any, obj: any) => any;\n\n/**\n * @summary Allows for the injection of an {@link injectable} decorated dependency\n * @description the property must be typed for the requested dependency.\n *\n * Only concrete classes. No generics are supported\n *\n * Injected properties should be described like so:\n * <pre>\n *     class ClassName {\n *         ...\n *\n *         @inject()\n *         propertyName!: InjectableClass;\n *\n *         ...\n *     }\n * </pre>\n *\n * where InjectableClass is the class you want to inject.\n * Notice the use of '!:' to ensure the transpiler the property will be set outside the constructor but will always be defined\n * For project where minification occours, you should use the category param to ensure the name is the same throughout\n *\n * @param {string} [category] defaults to the class Name. (Useful when minification occours and names are changed so we can no longer rely on the class name, or when we want to upcast the Object)\n * @param {InstanceTransformer} [transformer]\n *\n * @function inject\n *\n * @memberOf module:injectable-decorators.Decorators\n */\nexport const inject =\n  (category?: string, transformer?: InstanceTransformer) =>\n  (target: any, propertyKey?: any) => {\n    const values = new WeakMap();\n\n    const name: string | undefined =\n      category || getTypeFromDecorator(target, propertyKey);\n    if (!name) throw new Error(`Could not get Type from decorator`);\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECT),\n      {\n        injectable: name,\n      },\n      target,\n      propertyKey\n    );\n\n    Object.defineProperty(target, propertyKey, {\n      configurable: true,\n      get(this: any) {\n        const descriptor: PropertyDescriptor = Object.getOwnPropertyDescriptor(\n          target,\n          propertyKey\n        ) as PropertyDescriptor;\n        if (descriptor.configurable) {\n          Object.defineProperty(this, propertyKey, {\n            enumerable: true,\n            configurable: false,\n            get(this: any) {\n              let obj = values.get(this);\n              if (!obj) {\n                obj = Injectables.get(name);\n                if (!obj)\n                  throw new Error(\n                    `Could not get Injectable ${name} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`\n                  );\n                if (transformer)\n                  try {\n                    obj = transformer(obj, target);\n                  } catch (e) {\n                    console.error(e);\n                  }\n                values.set(this, obj);\n              }\n              return obj;\n            },\n          });\n          return this[propertyKey];\n        }\n      },\n    });\n  };\n"]}
151
+ export function inject(category, transformer) {
152
+ return (target, propertyKey) => {
153
+ const values = new WeakMap();
154
+ const name = category || getTypeFromDecorator(target, propertyKey);
155
+ if (!name)
156
+ throw new Error(`Could not get Type from decorator`);
157
+ Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECT), {
158
+ injectable: name,
159
+ }, target, propertyKey);
160
+ Object.defineProperty(target, propertyKey, {
161
+ configurable: true,
162
+ get() {
163
+ const descriptor = Object.getOwnPropertyDescriptor(target, propertyKey);
164
+ if (descriptor.configurable) {
165
+ Object.defineProperty(this, propertyKey, {
166
+ enumerable: true,
167
+ configurable: false,
168
+ get() {
169
+ let obj = values.get(this);
170
+ if (!obj) {
171
+ obj = Injectables.get(name);
172
+ if (!obj)
173
+ throw new Error(`Could not get Injectable ${name} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`);
174
+ if (transformer)
175
+ try {
176
+ obj = transformer(obj, target);
177
+ }
178
+ catch (e) {
179
+ console.error(e);
180
+ }
181
+ values.set(this, obj);
182
+ }
183
+ return obj;
184
+ },
185
+ });
186
+ return this[propertyKey];
187
+ }
188
+ },
189
+ });
190
+ };
191
+ }
192
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import { InjectablesKeys } from \"./constants\";\nimport { Injectables } from \"./Injectables\";\nimport { getTypeFromDecorator } from \"./utils\";\n\n/**\n * @description Generates a fully qualified reflection metadata key.\n * @summary Returns the reflection key for injectables by prefixing the provided key with the base reflection key.\n * @param {string} key The key to be prefixed\n * @return {string} The fully qualified reflection key\n * @function getInjectKey\n * @memberOf module:injectable-decorators\n */\nconst getInjectKey = (key: string) => InjectablesKeys.REFLECT + key;\n\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable singleton that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns a singleton instance.\n *\n * @param {string} [category] Defaults to the class name. Useful when minification occurs and names are changed,\n * or when you want to upcast the object to a different type.\n * @param {boolean} [force] Defines if the injectable should override an already existing instance (if any).\n * Only meant for extending decorators.\n * @param {Function} [instanceCallback] Optional callback function that will be called with the instance after creation.\n * @return {Function} A decorator function that transforms the class into an injectable.\n *\n * @function injectable\n * @category Class Decorators\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Decorator\n *   participant Injectables\n *\n *   Client->>Decorator: @injectable()\n *   Decorator->>Decorator: Create new constructor\n *\n *   Note over Decorator: When new instance requested\n *   Decorator->>Injectables: get(name)\n *   alt Instance exists\n *     Injectables-->>Decorator: Return existing instance\n *   else No instance\n *     Decorator->>Injectables: register(original, name)\n *     Decorator->>Injectables: get(name)\n *     Injectables-->>Decorator: Return new instance\n *     opt Has callback\n *       Decorator->>Decorator: Call instanceCallback\n *     end\n *   end\n *   Decorator->>Decorator: Define metadata\n *   Decorator-->>Client: Return instance\n */\nexport function injectable(\n  category: string | undefined = undefined,\n  force: boolean = false,\n  instanceCallback?: (instance: any, ...args: any[]) => void\n) {\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  return (original: any, propertyKey?: any) => {\n    const name = category || original.name;\n    // the new constructor behaviour\n    const newConstructor: any = function (...args: any[]) {\n      let inj: any = Injectables.get<any>(name, ...args);\n      if (!inj) {\n        Injectables.register(original, name, true, force);\n        inj = Injectables.get<any>(name, ...args);\n        if (!inj) return undefined;\n\n        if (instanceCallback)\n          try {\n            instanceCallback(inj);\n          } catch (e: any) {\n            console.error(\n              `Failed to call injectable callback for ${name}: ${e}`\n            );\n          }\n      }\n\n      const metadata = Object.assign(\n        {},\n        {\n          class: name,\n        }\n      );\n\n      Reflect.defineMetadata(\n        getInjectKey(InjectablesKeys.INJECTABLE),\n        metadata,\n        inj.constructor\n      );\n\n      return inj;\n    };\n\n    // copy prototype so instanceof operator still works\n    newConstructor.prototype = original.prototype;\n    // newConstructor.__proto__ = original.__proto__;\n    // Sets the proper constructor name for type verification\n    Object.defineProperty(newConstructor, \"name\", {\n      writable: false,\n      enumerable: true,\n      configurable: false,\n      value: original.prototype.constructor.name,\n    });\n    // return new constructor (will override original)\n    return newConstructor;\n  };\n}\n/**\n * @description Function type for transforming injectable instances before they're injected.\n * @summary Function which transforms a cached {@link injectable} instance before it's injected into a target object.\n *\n * @param {any} injectable The injectable instance to transform\n * @param {any} obj The object the injectable will be injected on\n * @return {any} The transformed injectable instance\n *\n * @typedef {Function} InstanceTransformer\n * @memberOf module:injectable-decorators\n */\nexport type InstanceTransformer = (injectable: any, obj: any) => any;\n\n/**\n * @description Property decorator that injects a dependency into a class property.\n * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.\n * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.\n *\n * Injected properties should be described like so:\n * <pre>\n *     class ClassName {\n *         ...\n *\n *         @inject()\n *         propertyName!: InjectableClass;\n *\n *         ...\n *     }\n * </pre>\n *\n * where InjectableClass is the class you want to inject.\n * Notice the use of '!:' to ensure the transpiler the property will be set outside the constructor but will always be defined.\n * For projects where minification occurs, you should use the category param to ensure the name is the same throughout.\n *\n * @param {string} [category] Defaults to the class name derived from the property type. Useful when minification occurs\n * and names are changed, or when you want to upcast the object to a different type.\n * @param {InstanceTransformer} [transformer] Optional function to transform the injectable instance before it's injected.\n * @return {Function} A property decorator function that sets up the dependency injection.\n *\n * @function inject\n * @category Property Decorators\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Decorator\n *   participant Injectables\n *\n *   Client->>Decorator: @inject()\n *   Decorator->>Decorator: Get type from property\n *   Decorator->>Decorator: Define metadata\n *   Decorator->>Decorator: Define property getter\n *\n *   Note over Decorator: When property accessed\n *   Client->>Decorator: access property\n *   Decorator->>Decorator: Check if instance exists\n *   alt Instance exists in WeakMap\n *     Decorator-->>Client: Return cached instance\n *   else No instance\n *     Decorator->>Injectables: get(name)\n *     alt Injectable found\n *       Injectables-->>Decorator: Return injectable instance\n *       opt Has transformer\n *         Decorator->>Decorator: Call transformer\n *       end\n *       Decorator->>Decorator: Store in WeakMap\n *       Decorator-->>Client: Return instance\n *     else No injectable\n *       Decorator-->>Client: Throw error\n *     end\n *   end\n */\nexport function inject(category?: string, transformer?: InstanceTransformer) {\n  return (target: any, propertyKey?: any) => {\n    const values = new WeakMap();\n\n    const name: string | undefined =\n      category || getTypeFromDecorator(target, propertyKey);\n    if (!name) throw new Error(`Could not get Type from decorator`);\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECT),\n      {\n        injectable: name,\n      },\n      target,\n      propertyKey\n    );\n\n    Object.defineProperty(target, propertyKey, {\n      configurable: true,\n      get(this: any) {\n        const descriptor: PropertyDescriptor = Object.getOwnPropertyDescriptor(\n          target,\n          propertyKey\n        ) as PropertyDescriptor;\n        if (descriptor.configurable) {\n          Object.defineProperty(this, propertyKey, {\n            enumerable: true,\n            configurable: false,\n            get(this: any) {\n              let obj = values.get(this);\n              if (!obj) {\n                obj = Injectables.get(name);\n                if (!obj)\n                  throw new Error(\n                    `Could not get Injectable ${name} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`\n                  );\n                if (transformer)\n                  try {\n                    obj = transformer(obj, target);\n                  } catch (e) {\n                    console.error(e);\n                  }\n                values.set(this, obj);\n              }\n              return obj;\n            },\n          });\n          return this[propertyKey];\n        }\n      },\n    });\n  };\n}\n"]}
@@ -1,22 +1,20 @@
1
1
  /**
2
+ * @description A lightweight dependency injection library for TypeScript applications.
2
3
  * @summary Adds a simple Injectables implementation to create singleton instances of an object
3
- * and easily inject it into other objects
4
+ * and easily inject it into other objects. Provides decorators for marking classes as injectable
5
+ * and for injecting dependencies into class properties.
4
6
  *
5
7
  * @module injectable-decorators
6
8
  */
7
- /**
8
- * @summary functions that decorate classes or class properties
9
- * @namespace Decorators
10
- * @memberOf module:injectable-decorators
11
- */
12
9
  export * from "./constants";
13
10
  export * from "./decorators";
14
11
  export * from "./Injectables";
15
12
  export * from "./registry";
16
13
  export * from "./utils";
17
14
  /**
18
- * @summary Defined on library build. holds the library current version
15
+ * @description Current version of the injectable-decorators library.
16
+ * @summary Defined on library build. Holds the library's current version string.
19
17
  * @const VERSION
20
18
  * @memberOf module:injectable-decorators
21
19
  */
22
- export declare const VERSION = "1.6.1";
20
+ export declare const VERSION = "1.6.2";
package/lib/esm/index.js CHANGED
@@ -1,23 +1,21 @@
1
1
  /**
2
+ * @description A lightweight dependency injection library for TypeScript applications.
2
3
  * @summary Adds a simple Injectables implementation to create singleton instances of an object
3
- * and easily inject it into other objects
4
+ * and easily inject it into other objects. Provides decorators for marking classes as injectable
5
+ * and for injecting dependencies into class properties.
4
6
  *
5
7
  * @module injectable-decorators
6
8
  */
9
+ export * from "./constants.js";
10
+ export * from "./decorators.js";
11
+ export * from "./Injectables.js";
12
+ export * from "./registry.js";
13
+ export * from "./utils.js";
7
14
  /**
8
- * @summary functions that decorate classes or class properties
9
- * @namespace Decorators
10
- * @memberOf module:injectable-decorators
11
- */
12
- export * from "./constants";
13
- export * from "./decorators";
14
- export * from "./Injectables";
15
- export * from "./registry";
16
- export * from "./utils";
17
- /**
18
- * @summary Defined on library build. holds the library current version
15
+ * @description Current version of the injectable-decorators library.
16
+ * @summary Defined on library build. Holds the library's current version string.
19
17
  * @const VERSION
20
18
  * @memberOf module:injectable-decorators
21
19
  */
22
- export const VERSION = "1.6.1";
23
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7O0dBS0c7QUFFSDs7OztHQUlHO0FBRUgsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxTQUFTLENBQUM7QUFFeEI7Ozs7R0FJRztBQUNILE1BQU0sQ0FBQyxNQUFNLE9BQU8sR0FBRyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBzdW1tYXJ5IEFkZHMgYSBzaW1wbGUgSW5qZWN0YWJsZXMgaW1wbGVtZW50YXRpb24gdG8gY3JlYXRlIHNpbmdsZXRvbiBpbnN0YW5jZXMgb2YgYW4gb2JqZWN0XG4gKiBhbmQgZWFzaWx5IGluamVjdCBpdCBpbnRvIG90aGVyIG9iamVjdHNcbiAqXG4gKiBAbW9kdWxlIGluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5cbi8qKlxuICogQHN1bW1hcnkgZnVuY3Rpb25zIHRoYXQgZGVjb3JhdGUgY2xhc3NlcyBvciBjbGFzcyBwcm9wZXJ0aWVzXG4gKiBAbmFtZXNwYWNlIERlY29yYXRvcnNcbiAqIEBtZW1iZXJPZiBtb2R1bGU6aW5qZWN0YWJsZS1kZWNvcmF0b3JzXG4gKi9cblxuZXhwb3J0ICogZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kZWNvcmF0b3JzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9JbmplY3RhYmxlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vcmVnaXN0cnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG5cbi8qKlxuICogQHN1bW1hcnkgRGVmaW5lZCBvbiBsaWJyYXJ5IGJ1aWxkLiBob2xkcyB0aGUgbGlicmFyeSBjdXJyZW50IHZlcnNpb25cbiAqIEBjb25zdCBWRVJTSU9OXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbiJdfQ==
20
+ export const VERSION = "1.6.2";
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7R0FPRztBQUVILCtCQUE0QjtBQUM1QixnQ0FBNkI7QUFDN0IsaUNBQThCO0FBQzlCLDhCQUEyQjtBQUMzQiwyQkFBd0I7QUFFeEI7Ozs7O0dBS0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsYUFBYSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAZGVzY3JpcHRpb24gQSBsaWdodHdlaWdodCBkZXBlbmRlbmN5IGluamVjdGlvbiBsaWJyYXJ5IGZvciBUeXBlU2NyaXB0IGFwcGxpY2F0aW9ucy5cbiAqIEBzdW1tYXJ5IEFkZHMgYSBzaW1wbGUgSW5qZWN0YWJsZXMgaW1wbGVtZW50YXRpb24gdG8gY3JlYXRlIHNpbmdsZXRvbiBpbnN0YW5jZXMgb2YgYW4gb2JqZWN0XG4gKiBhbmQgZWFzaWx5IGluamVjdCBpdCBpbnRvIG90aGVyIG9iamVjdHMuIFByb3ZpZGVzIGRlY29yYXRvcnMgZm9yIG1hcmtpbmcgY2xhc3NlcyBhcyBpbmplY3RhYmxlXG4gKiBhbmQgZm9yIGluamVjdGluZyBkZXBlbmRlbmNpZXMgaW50byBjbGFzcyBwcm9wZXJ0aWVzLlxuICpcbiAqIEBtb2R1bGUgaW5qZWN0YWJsZS1kZWNvcmF0b3JzXG4gKi9cblxuZXhwb3J0ICogZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kZWNvcmF0b3JzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9JbmplY3RhYmxlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vcmVnaXN0cnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgaW5qZWN0YWJsZS1kZWNvcmF0b3JzIGxpYnJhcnkuXG4gKiBAc3VtbWFyeSBEZWZpbmVkIG9uIGxpYnJhcnkgYnVpbGQuIEhvbGRzIHRoZSBsaWJyYXJ5J3MgY3VycmVudCB2ZXJzaW9uIHN0cmluZy5cbiAqIEBjb25zdCBWRVJTSU9OXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbiJdfQ==
@@ -1,46 +1,93 @@
1
1
  /**
2
- * @summary defines an Injectable type
2
+ * @description Type representing either a class constructor or an instance.
3
+ * @summary Defines an Injectable type that can be either a class constructor or an instance of a class.
4
+ * @template T The type of the injectable object
5
+ * @typedef {function(any): T | T} Injectable
3
6
  * @memberOf module:injectable-decorators
4
7
  */
5
8
  export type Injectable<T> = {
6
9
  new (...args: any[]): T;
7
10
  } | T;
8
11
  /**
9
- * @summary Interface for an injectable registry
10
- * @interface InjectableRegistry
12
+ * @description Contract for a registry that manages injectable objects.
13
+ * @summary Interface for an injectable registry that provides methods for retrieving, registering, and building injectable objects.
14
+ * @template T Type parameter used in the interface methods
15
+ * @interface InjectablesRegistry
16
+ * @memberOf module:injectable-decorators
11
17
  */
12
18
  export interface InjectablesRegistry {
13
19
  /**
14
- * @summary retrieves an {@link Injectable}
15
- * @param {string} name
16
- * @param {any[]} args
17
- * @return {Injectable | undefined}
18
- *
19
- * @method
20
+ * @description Fetches an injectable instance by its registered name.
21
+ * @summary Retrieves an {@link Injectable} from the registry by name, optionally passing constructor arguments.
22
+ * @template T Type of the injectable object to retrieve
23
+ * @param {string} name The registered name of the injectable to retrieve
24
+ * @param {any[]} args Constructor arguments to pass when instantiating the injectable
25
+ * @return {Injectable<T> | undefined} The injectable instance or undefined if not found
26
+ * @memberOf module:injectable-decorators
20
27
  */
21
28
  get<T>(name: string, ...args: any[]): Injectable<T> | undefined;
22
29
  /**
23
- * @summary registers an injectable constructor
24
- * @param {Injectable} constructor
25
- * @param {any[]} args
26
- *
27
- * @method
30
+ * @description Adds a class or object to the injectable registry.
31
+ * @summary Registers an injectable constructor or instance with the registry, making it available for injection.
32
+ * @template T Type of the injectable object to register
33
+ * @param {Injectable<T>} constructor The class constructor or object instance to register
34
+ * @param {any[]} args Additional arguments for registration (category, singleton flag, etc.)
35
+ * @return {void}
36
+ * @memberOf module:injectable-decorators
28
37
  */
29
38
  register<T>(constructor: Injectable<T>, ...args: any[]): void;
30
39
  /**
31
- * @summary Instantiates an Injectable
32
- * @param {Record<string, any>} obj
33
- * @param {any[]} args
34
- * @return T
35
- *
36
- * @method
40
+ * @description Creates a new instance of an injectable class.
41
+ * @summary Instantiates an injectable class using its constructor and the provided arguments.
42
+ * @template T Type of the object to build
43
+ * @param {Record<string, any>} obj Object containing the name of the injectable to build
44
+ * @param {any[]} args Constructor arguments to pass when instantiating the injectable
45
+ * @return {T} The newly created instance
46
+ * @memberOf module:injectable-decorators
37
47
  */
38
48
  build<T>(obj: Record<string, any>, ...args: any[]): T;
39
49
  }
40
50
  /**
41
- * @summary Holds the vairous {@link Injectable}s
51
+ * @description Default implementation of the InjectablesRegistry interface.
52
+ * @summary Holds the various {@link Injectable}s in a cache and provides methods to register, retrieve, and build them.
53
+ * @template T Type parameter used in the class methods
54
+ *
42
55
  * @class InjectableRegistryImp
43
56
  * @implements InjectablesRegistry
57
+ *
58
+ * @memberOf module:injectable-decorators
59
+ *
60
+ * @example
61
+ * // Create a new registry
62
+ * const registry = new InjectableRegistryImp();
63
+ *
64
+ * // Register a class
65
+ * class MyService {
66
+ * doSomething() {
67
+ * return 'Hello World';
68
+ * }
69
+ * }
70
+ * registry.register(MyService, 'MyService', true);
71
+ *
72
+ * // Get the instance
73
+ * const service = registry.get('MyService');
74
+ * service.doSomething(); // 'Hello World'
75
+ *
76
+ * @mermaid
77
+ * sequenceDiagram
78
+ * participant Client
79
+ * participant Registry
80
+ *
81
+ * Client->>Registry: register(MyService)
82
+ * Registry->>Registry: Store in cache
83
+ *
84
+ * Client->>Registry: get("MyService")
85
+ * alt Instance exists and is singleton
86
+ * Registry-->>Client: Return cached instance
87
+ * else No instance or not singleton
88
+ * Registry->>Registry: build(name)
89
+ * Registry-->>Client: Return new instance
90
+ * end
44
91
  */
45
92
  export declare class InjectableRegistryImp implements InjectablesRegistry {
46
93
  private cache;
@@ -1,7 +1,44 @@
1
1
  /**
2
- * @summary Holds the vairous {@link Injectable}s
2
+ * @description Default implementation of the InjectablesRegistry interface.
3
+ * @summary Holds the various {@link Injectable}s in a cache and provides methods to register, retrieve, and build them.
4
+ * @template T Type parameter used in the class methods
5
+ *
3
6
  * @class InjectableRegistryImp
4
7
  * @implements InjectablesRegistry
8
+ *
9
+ * @memberOf module:injectable-decorators
10
+ *
11
+ * @example
12
+ * // Create a new registry
13
+ * const registry = new InjectableRegistryImp();
14
+ *
15
+ * // Register a class
16
+ * class MyService {
17
+ * doSomething() {
18
+ * return 'Hello World';
19
+ * }
20
+ * }
21
+ * registry.register(MyService, 'MyService', true);
22
+ *
23
+ * // Get the instance
24
+ * const service = registry.get('MyService');
25
+ * service.doSomething(); // 'Hello World'
26
+ *
27
+ * @mermaid
28
+ * sequenceDiagram
29
+ * participant Client
30
+ * participant Registry
31
+ *
32
+ * Client->>Registry: register(MyService)
33
+ * Registry->>Registry: Store in cache
34
+ *
35
+ * Client->>Registry: get("MyService")
36
+ * alt Instance exists and is singleton
37
+ * Registry-->>Client: Return cached instance
38
+ * else No instance or not singleton
39
+ * Registry->>Registry: build(name)
40
+ * Registry-->>Client: Return new instance
41
+ * end
5
42
  */
6
43
  export class InjectableRegistryImp {
7
44
  constructor() {
@@ -56,4 +93,4 @@ export class InjectableRegistryImp {
56
93
  return instance;
57
94
  }
58
95
  }
59
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVnaXN0cnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcmVnaXN0cnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBeUNBOzs7O0dBSUc7QUFDSCxNQUFNLE9BQU8scUJBQXFCO0lBQWxDO1FBQ1UsVUFBSyxHQUErQixFQUFFLENBQUM7SUE0RGpELENBQUM7SUExREM7O09BRUc7SUFDSCxHQUFHLENBQUksSUFBWSxFQUFFLEdBQUcsSUFBVztRQUNqQyxJQUFJLENBQUM7WUFDSCxNQUFNLFVBQVUsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1lBQ3BDLE1BQU0sUUFBUSxHQUFHLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxDQUFDO1lBQ2hDLElBQUksQ0FBQyxVQUFVLENBQUMsU0FBUyxJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVE7Z0JBQy9DLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBSSxRQUFRLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQztZQUMxQyxPQUFPLFVBQVUsQ0FBQyxRQUFRLElBQUksSUFBSSxDQUFDLEtBQUssQ0FBSSxRQUFRLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQztZQUMvRCw2REFBNkQ7UUFDL0QsQ0FBQztRQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDWCxPQUFPLFNBQVMsQ0FBQztRQUNuQixDQUFDO0lBQ0gsQ0FBQztJQUNEOztPQUVHO0lBQ0gsUUFBUSxDQUNOLEdBQWtCLEVBQ2xCLFdBQStCLFNBQVMsRUFDeEMsY0FBdUIsSUFBSSxFQUMzQixRQUFpQixLQUFLO1FBRXRCLE1BQU0sT0FBTyxHQUF3QixHQUEwQixDQUFDO1FBRWhFLE1BQU0sV0FBVyxHQUFHLENBQUMsT0FBTyxDQUFDLElBQUksSUFBSSxPQUFPLENBQUMsV0FBVyxDQUFDO1FBQ3pELElBQUksT0FBTyxPQUFPLEtBQUssVUFBVSxJQUFJLENBQUMsV0FBVztZQUMvQyxNQUFNLElBQUksS0FBSyxDQUNiLGtFQUFrRSxDQUNuRSxDQUFDO1FBRUosTUFBTSxJQUFJLEdBQ1IsUUFBUTtZQUNSLENBQUMsV0FBVyxJQUFJLFdBQVcsQ0FBQyxJQUFJLElBQUksV0FBVyxDQUFDLElBQUksS0FBSyxVQUFVO2dCQUNqRSxDQUFDLENBQUUsV0FBMEMsQ0FBQyxJQUFJO2dCQUNsRCxDQUFDLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXBCLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLEtBQUs7WUFDNUIsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsR0FBRztnQkFDakIsUUFBUSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxTQUFTO2dCQUN2QyxXQUFXLEVBQUUsQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsU0FBUztnQkFDM0MsU0FBUyxFQUFFLFdBQVc7YUFDdkIsQ0FBQztJQUNOLENBQUM7SUFDRDs7T0FFRztJQUNILEtBQUssQ0FBSSxJQUFzQixFQUFFLEdBQUcsSUFBVztRQUM3QyxNQUFNLEVBQUUsV0FBVyxFQUFFLFNBQVMsRUFBRSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3pELE1BQU0sUUFBUSxHQUFHLElBQUksV0FBVyxDQUFDLEdBQUcsSUFBSSxDQUFDLENBQUM7UUFDMUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUc7WUFDdEIsUUFBUSxFQUFFLFFBQVE7WUFDbEIsV0FBVyxFQUFFLFdBQVc7WUFDeEIsU0FBUyxFQUFFLFNBQVM7U0FDckIsQ0FBQztRQUNGLE9BQU8sUUFBUSxDQUFDO0lBQ2xCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQHN1bW1hcnkgZGVmaW5lcyBhbiBJbmplY3RhYmxlIHR5cGVcbiAqIEBtZW1iZXJPZiBtb2R1bGU6aW5qZWN0YWJsZS1kZWNvcmF0b3JzXG4gKi9cbmV4cG9ydCB0eXBlIEluamVjdGFibGU8VD4gPSB7IG5ldyAoLi4uYXJnczogYW55W10pOiBUIH0gfCBUO1xuXG4vKipcbiAqIEBzdW1tYXJ5IEludGVyZmFjZSBmb3IgYW4gaW5qZWN0YWJsZSByZWdpc3RyeVxuICogQGludGVyZmFjZSBJbmplY3RhYmxlUmVnaXN0cnlcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBJbmplY3RhYmxlc1JlZ2lzdHJ5IHtcbiAgLyoqXG4gICAqIEBzdW1tYXJ5IHJldHJpZXZlcyBhbiB7QGxpbmsgSW5qZWN0YWJsZX1cbiAgICogQHBhcmFtIHtzdHJpbmd9IG5hbWVcbiAgICogQHBhcmFtIHthbnlbXX0gYXJnc1xuICAgKiBAcmV0dXJuIHtJbmplY3RhYmxlIHwgdW5kZWZpbmVkfVxuICAgKlxuICAgKiBAbWV0aG9kXG4gICAqL1xuICBnZXQ8VD4obmFtZTogc3RyaW5nLCAuLi5hcmdzOiBhbnlbXSk6IEluamVjdGFibGU8VD4gfCB1bmRlZmluZWQ7XG5cbiAgLyoqXG4gICAqIEBzdW1tYXJ5IHJlZ2lzdGVycyBhbiBpbmplY3RhYmxlIGNvbnN0cnVjdG9yXG4gICAqIEBwYXJhbSB7SW5qZWN0YWJsZX0gY29uc3RydWN0b3JcbiAgICogQHBhcmFtIHthbnlbXX0gYXJnc1xuICAgKlxuICAgKiBAbWV0aG9kXG4gICAqL1xuICByZWdpc3RlcjxUPihjb25zdHJ1Y3RvcjogSW5qZWN0YWJsZTxUPiwgLi4uYXJnczogYW55W10pOiB2b2lkO1xuXG4gIC8qKlxuICAgKiBAc3VtbWFyeSBJbnN0YW50aWF0ZXMgYW4gSW5qZWN0YWJsZVxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IG9ialxuICAgKiBAcGFyYW0ge2FueVtdfSBhcmdzXG4gICAqIEByZXR1cm4gVFxuICAgKlxuICAgKiBAbWV0aG9kXG4gICAqL1xuICBidWlsZDxUPihvYmo6IFJlY29yZDxzdHJpbmcsIGFueT4sIC4uLmFyZ3M6IGFueVtdKTogVDtcbn1cblxuLyoqXG4gKiBAc3VtbWFyeSBIb2xkcyB0aGUgdmFpcm91cyB7QGxpbmsgSW5qZWN0YWJsZX1zXG4gKiBAY2xhc3MgSW5qZWN0YWJsZVJlZ2lzdHJ5SW1wXG4gKiBAaW1wbGVtZW50cyBJbmplY3RhYmxlc1JlZ2lzdHJ5XG4gKi9cbmV4cG9ydCBjbGFzcyBJbmplY3RhYmxlUmVnaXN0cnlJbXAgaW1wbGVtZW50cyBJbmplY3RhYmxlc1JlZ2lzdHJ5IHtcbiAgcHJpdmF0ZSBjYWNoZTogeyBbaW5kZXhlcjogc3RyaW5nXTogYW55IH0gPSB7fTtcblxuICAvKipcbiAgICogQGluaGVyaXREb2NcbiAgICovXG4gIGdldDxUPihuYW1lOiBzdHJpbmcsIC4uLmFyZ3M6IGFueVtdKTogVCB8IHVuZGVmaW5lZCB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IGlubmVyQ2FjaGUgPSB0aGlzLmNhY2hlW25hbWVdO1xuICAgICAgY29uc3QgYnVpbGREZWYgPSB7IG5hbWU6IG5hbWUgfTtcbiAgICAgIGlmICghaW5uZXJDYWNoZS5zaW5nbGV0b24gJiYgIWlubmVyQ2FjaGUuaW5zdGFuY2UpXG4gICAgICAgIHJldHVybiB0aGlzLmJ1aWxkPFQ+KGJ1aWxkRGVmLCAuLi5hcmdzKTtcbiAgICAgIHJldHVybiBpbm5lckNhY2hlLmluc3RhbmNlIHx8IHRoaXMuYnVpbGQ8VD4oYnVpbGREZWYsIC4uLmFyZ3MpO1xuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgfVxuICB9XG4gIC8qKlxuICAgKiBAaW5oZXJpdERvY1xuICAgKi9cbiAgcmVnaXN0ZXI8VD4oXG4gICAgb2JqOiBJbmplY3RhYmxlPFQ+LFxuICAgIGNhdGVnb3J5OiBzdHJpbmcgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQsXG4gICAgaXNTaW5nbGV0b246IGJvb2xlYW4gPSB0cnVlLFxuICAgIGZvcmNlOiBib29sZWFuID0gZmFsc2VcbiAgKTogdm9pZCB7XG4gICAgY29uc3QgY2FzdE9iajogUmVjb3JkPHN0cmluZywgYW55PiA9IG9iaiBhcyBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuXG4gICAgY29uc3QgY29uc3RydWN0b3IgPSAhY2FzdE9iai5uYW1lICYmIGNhc3RPYmouY29uc3RydWN0b3I7XG4gICAgaWYgKHR5cGVvZiBjYXN0T2JqICE9PSBcImZ1bmN0aW9uXCIgJiYgIWNvbnN0cnVjdG9yKVxuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgSW5qZWN0YWJsZSByZWdpc3RlcmluZyBmYWlsZWQuIE1pc3NpbmcgQ2xhc3MgbmFtZSBvciBjb25zdHJ1Y3RvcmBcbiAgICAgICk7XG5cbiAgICBjb25zdCBuYW1lID1cbiAgICAgIGNhdGVnb3J5IHx8XG4gICAgICAoY29uc3RydWN0b3IgJiYgY29uc3RydWN0b3IubmFtZSAmJiBjb25zdHJ1Y3Rvci5uYW1lICE9PSBcIkZ1bmN0aW9uXCJcbiAgICAgICAgPyAoY29uc3RydWN0b3IgYXMgeyBbaW5kZXhlcjogc3RyaW5nXTogYW55IH0pLm5hbWVcbiAgICAgICAgOiBjYXN0T2JqLm5hbWUpO1xuXG4gICAgaWYgKCF0aGlzLmNhY2hlW25hbWVdIHx8IGZvcmNlKVxuICAgICAgdGhpcy5jYWNoZVtuYW1lXSA9IHtcbiAgICAgICAgaW5zdGFuY2U6IGNvbnN0cnVjdG9yID8gb2JqIDogdW5kZWZpbmVkLFxuICAgICAgICBjb25zdHJ1Y3RvcjogIWNvbnN0cnVjdG9yID8gb2JqIDogdW5kZWZpbmVkLFxuICAgICAgICBzaW5nbGV0b246IGlzU2luZ2xldG9uLFxuICAgICAgfTtcbiAgfVxuICAvKipcbiAgICogQGluaGVyaXREb2NcbiAgICovXG4gIGJ1aWxkPFQ+KGRlZnM6IHsgbmFtZTogc3RyaW5nIH0sIC4uLmFyZ3M6IGFueVtdKTogVCB7XG4gICAgY29uc3QgeyBjb25zdHJ1Y3Rvciwgc2luZ2xldG9uIH0gPSB0aGlzLmNhY2hlW2RlZnMubmFtZV07XG4gICAgY29uc3QgaW5zdGFuY2UgPSBuZXcgY29uc3RydWN0b3IoLi4uYXJncyk7XG4gICAgdGhpcy5jYWNoZVtkZWZzLm5hbWVdID0ge1xuICAgICAgaW5zdGFuY2U6IGluc3RhbmNlLFxuICAgICAgY29uc3RydWN0b3I6IGNvbnN0cnVjdG9yLFxuICAgICAgc2luZ2xldG9uOiBzaW5nbGV0b24sXG4gICAgfTtcbiAgICByZXR1cm4gaW5zdGFuY2U7XG4gIH1cbn1cbiJdfQ==
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/registry.ts"],"names":[],"mappings":"AAmDA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAyCG;AACH,MAAM,OAAO,qBAAqB;IAAlC;QACU,UAAK,GAA+B,EAAE,CAAC;IA4DjD,CAAC;IA1DC;;OAEG;IACH,GAAG,CAAI,IAAY,EAAE,GAAG,IAAW;QACjC,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YACpC,MAAM,QAAQ,GAAG,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC;YAChC,IAAI,CAAC,UAAU,CAAC,SAAS,IAAI,CAAC,UAAU,CAAC,QAAQ;gBAC/C,OAAO,IAAI,CAAC,KAAK,CAAI,QAAQ,EAAE,GAAG,IAAI,CAAC,CAAC;YAC1C,OAAO,UAAU,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,CAAI,QAAQ,EAAE,GAAG,IAAI,CAAC,CAAC;YAC/D,6DAA6D;QAC/D,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,SAAS,CAAC;QACnB,CAAC;IACH,CAAC;IACD;;OAEG;IACH,QAAQ,CACN,GAAkB,EAClB,WAA+B,SAAS,EACxC,cAAuB,IAAI,EAC3B,QAAiB,KAAK;QAEtB,MAAM,OAAO,GAAwB,GAA0B,CAAC;QAEhE,MAAM,WAAW,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,OAAO,CAAC,WAAW,CAAC;QACzD,IAAI,OAAO,OAAO,KAAK,UAAU,IAAI,CAAC,WAAW;YAC/C,MAAM,IAAI,KAAK,CACb,kEAAkE,CACnE,CAAC;QAEJ,MAAM,IAAI,GACR,QAAQ;YACR,CAAC,WAAW,IAAI,WAAW,CAAC,IAAI,IAAI,WAAW,CAAC,IAAI,KAAK,UAAU;gBACjE,CAAC,CAAE,WAA0C,CAAC,IAAI;gBAClD,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAEpB,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK;YAC5B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG;gBACjB,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS;gBACvC,WAAW,EAAE,CAAC,WAAW,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS;gBAC3C,SAAS,EAAE,WAAW;aACvB,CAAC;IACN,CAAC;IACD;;OAEG;IACH,KAAK,CAAI,IAAsB,EAAE,GAAG,IAAW;QAC7C,MAAM,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzD,MAAM,QAAQ,GAAG,IAAI,WAAW,CAAC,GAAG,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG;YACtB,QAAQ,EAAE,QAAQ;YAClB,WAAW,EAAE,WAAW;YACxB,SAAS,EAAE,SAAS;SACrB,CAAC;QACF,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF","sourcesContent":["/**\n * @description Type representing either a class constructor or an instance.\n * @summary Defines an Injectable type that can be either a class constructor or an instance of a class.\n * @template T The type of the injectable object\n * @typedef {function(any): T | T} Injectable\n * @memberOf module:injectable-decorators\n */\nexport type Injectable<T> = { new (...args: any[]): T } | T;\n\n/**\n * @description Contract for a registry that manages injectable objects.\n * @summary Interface for an injectable registry that provides methods for retrieving, registering, and building injectable objects.\n * @template T Type parameter used in the interface methods\n * @interface InjectablesRegistry\n * @memberOf module:injectable-decorators\n */\nexport interface InjectablesRegistry {\n  /**\n   * @description Fetches an injectable instance by its registered name.\n   * @summary Retrieves an {@link Injectable} from the registry by name, optionally passing constructor arguments.\n   * @template T Type of the injectable object to retrieve\n   * @param {string} name The registered name of the injectable to retrieve\n   * @param {any[]} args Constructor arguments to pass when instantiating the injectable\n   * @return {Injectable<T> | undefined} The injectable instance or undefined if not found\n   * @memberOf module:injectable-decorators\n   */\n  get<T>(name: string, ...args: any[]): Injectable<T> | undefined;\n\n  /**\n   * @description Adds a class or object to the injectable registry.\n   * @summary Registers an injectable constructor or instance with the registry, making it available for injection.\n   * @template T Type of the injectable object to register\n   * @param {Injectable<T>} constructor The class constructor or object instance to register\n   * @param {any[]} args Additional arguments for registration (category, singleton flag, etc.)\n   * @return {void}\n   * @memberOf module:injectable-decorators\n   */\n  register<T>(constructor: Injectable<T>, ...args: any[]): void;\n\n  /**\n   * @description Creates a new instance of an injectable class.\n   * @summary Instantiates an injectable class using its constructor and the provided arguments.\n   * @template T Type of the object to build\n   * @param {Record<string, any>} obj Object containing the name of the injectable to build\n   * @param {any[]} args Constructor arguments to pass when instantiating the injectable\n   * @return {T} The newly created instance\n   * @memberOf module:injectable-decorators\n   */\n  build<T>(obj: Record<string, any>, ...args: any[]): T;\n}\n\n/**\n * @description Default implementation of the InjectablesRegistry interface.\n * @summary Holds the various {@link Injectable}s in a cache and provides methods to register, retrieve, and build them.\n * @template T Type parameter used in the class methods\n *\n * @class InjectableRegistryImp\n * @implements InjectablesRegistry\n *\n * @memberOf module:injectable-decorators\n *\n * @example\n * // Create a new registry\n * const registry = new InjectableRegistryImp();\n *\n * // Register a class\n * class MyService {\n *   doSomething() {\n *     return 'Hello World';\n *   }\n * }\n * registry.register(MyService, 'MyService', true);\n *\n * // Get the instance\n * const service = registry.get('MyService');\n * service.doSomething(); // 'Hello World'\n *\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Registry\n *\n *   Client->>Registry: register(MyService)\n *   Registry->>Registry: Store in cache\n *\n *   Client->>Registry: get(\"MyService\")\n *   alt Instance exists and is singleton\n *     Registry-->>Client: Return cached instance\n *   else No instance or not singleton\n *     Registry->>Registry: build(name)\n *     Registry-->>Client: Return new instance\n *   end\n */\nexport class InjectableRegistryImp implements InjectablesRegistry {\n  private cache: { [indexer: string]: any } = {};\n\n  /**\n   * @inheritDoc\n   */\n  get<T>(name: string, ...args: any[]): T | undefined {\n    try {\n      const innerCache = this.cache[name];\n      const buildDef = { name: name };\n      if (!innerCache.singleton && !innerCache.instance)\n        return this.build<T>(buildDef, ...args);\n      return innerCache.instance || this.build<T>(buildDef, ...args);\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    } catch (e) {\n      return undefined;\n    }\n  }\n  /**\n   * @inheritDoc\n   */\n  register<T>(\n    obj: Injectable<T>,\n    category: string | undefined = undefined,\n    isSingleton: boolean = true,\n    force: boolean = false\n  ): void {\n    const castObj: Record<string, any> = obj as Record<string, any>;\n\n    const constructor = !castObj.name && castObj.constructor;\n    if (typeof castObj !== \"function\" && !constructor)\n      throw new Error(\n        `Injectable registering failed. Missing Class name or constructor`\n      );\n\n    const name =\n      category ||\n      (constructor && constructor.name && constructor.name !== \"Function\"\n        ? (constructor as { [indexer: string]: any }).name\n        : castObj.name);\n\n    if (!this.cache[name] || force)\n      this.cache[name] = {\n        instance: constructor ? obj : undefined,\n        constructor: !constructor ? obj : undefined,\n        singleton: isSingleton,\n      };\n  }\n  /**\n   * @inheritDoc\n   */\n  build<T>(defs: { name: string }, ...args: any[]): T {\n    const { constructor, singleton } = this.cache[defs.name];\n    const instance = new constructor(...args);\n    this.cache[defs.name] = {\n      instance: instance,\n      constructor: constructor,\n      singleton: singleton,\n    };\n    return instance;\n  }\n}\n"]}