@decaf-ts/injectable-decorators 1.6.7 → 1.6.9

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,60 +1,101 @@
1
1
  import { InstanceCallback } from "./types";
2
2
  /**
3
- * @description Generates a fully qualified reflection metadata key.
4
- * @summary Returns the reflection key for injectables by prefixing the provided key with the base reflection key.
5
- * @param {string} key The key to be prefixed
6
- * @return {string} The fully qualified reflection key
7
- * @function getInjectKey
3
+ * @description Configuration options for the @injectable decorator.
4
+ * @summary Controls lifecycle (singleton vs on-demand) and an optional instance transformation callback executed post-construction.
5
+ * @template T The instance type affected by the callback when provided.
6
+ * @property {boolean} singleton When true, a single instance is shared (singleton). When false, instances are created on demand.
7
+ * @property {InstanceCallback<T>} [callback] Optional hook to transform the instance after it is constructed.
8
+ * @typedef InjectableConfig
8
9
  * @memberOf module:injectable-decorators
9
10
  */
10
- export declare const getInjectKey: (key: string) => string;
11
+ export type InjectableConfig = {
12
+ singleton: boolean;
13
+ callback?: InstanceCallback<any>;
14
+ };
15
+ /**
16
+ * @description Generic constructor type for class-like values.
17
+ * @summary Represents any class that can be instantiated with arbitrary arguments, producing an instance of type T.
18
+ * @template T The instance type created by the constructor.
19
+ * @typedef Constructor
20
+ * @example
21
+ * // Using Constructor to type a factory
22
+ * function make<T>(Ctor: Constructor<T>, ...args: any[]): T {
23
+ * return new Ctor(...args);
24
+ * }
25
+ * @memberOf module:injectable-decorators
26
+ */
27
+ export type Constructor<T = any> = {
28
+ new (...args: any[]): T;
29
+ };
11
30
  /**
12
31
  * @description Decorator that marks a class as available for dependency injection.
13
- * @summary Defines a class as an injectable singleton that can be retrieved from the registry.
14
- * When applied to a class, replaces its constructor with one that returns a singleton instance.
32
+ * @summary Defines a class as an injectable that can be retrieved from the registry.
33
+ * When applied to a class, replaces its constructor with one that returns an instance.
34
+ *
35
+ * @return {function(any): any} A decorator function that transforms the class into an injectable.
15
36
  *
16
- * @param {string} [category] Defaults to the class name. Useful when minification occurs and names are changed,
37
+ * @function injectable
38
+ */
39
+ export declare function injectable(): (original: any) => any;
40
+ /**
41
+ * @description Decorator that marks a class as available for dependency injection.
42
+ * @summary Defines a class as an injectable that can be retrieved from the registry.
43
+ * When applied to a class, replaces its constructor with one that returns an instance.
44
+ *
45
+ * @param {string | Constructor} category Defaults to the class category. Useful when minification occurs and names are changed,
17
46
  * or when you want to upcast the object to a different type.
18
- * @param {boolean} [force] Defines if the injectable should override an already existing instance (if any).
19
- * Only meant for extending decorators.
20
- * @param {Function} [instanceCallback] Optional callback function that will be called with the instance after creation.
21
- * @return {Function} A decorator function that transforms the class into an injectable.
47
+ *
48
+ * @return {function(any): any} A decorator function that transforms the class into an injectable.
22
49
  *
23
50
  * @function injectable
24
- * @category Class Decorators
51
+ */
52
+ export declare function injectable(category: string | Constructor): (original: any) => any;
53
+ /**
54
+ * @description Decorator that marks a class as available for dependency injection.
55
+ * @summary Defines a class as an injectable that can be retrieved from the registry.
56
+ * When applied to a class, replaces its constructor with one that returns an instance.
25
57
  *
26
- * @mermaid
27
- * sequenceDiagram
28
- * participant Client
29
- * participant Decorator
30
- * participant Injectables
31
- *
32
- * Client->>Decorator: @injectable()
33
- * Decorator->>Decorator: Create new constructor
34
- *
35
- * Note over Decorator: When new instance requested
36
- * Decorator->>Injectables: get(name)
37
- * alt Instance exists
38
- * Injectables-->>Decorator: Return existing instance
39
- * else No instance
40
- * Decorator->>Injectables: register(original, name)
41
- * Decorator->>Injectables: get(name)
42
- * Injectables-->>Decorator: Return new instance
43
- * opt Has callback
44
- * Decorator->>Decorator: Call instanceCallback
45
- * end
46
- * end
47
- * Decorator->>Decorator: Define metadata
48
- * Decorator-->>Client: Return instance
58
+ * @param {Partial<InjectableConfig>} cfg=DefaultInjectableConfig Allows overriding the default singleton behavior and adding a callback function.
59
+ *
60
+ * @return {function(any): any} A decorator function that transforms the class into an injectable.
61
+ *
62
+ * @function injectable
49
63
  */
50
- export declare function injectable(): (original: any) => any;
51
- export declare function injectable(category: string | {
52
- new (...args: any[]): any;
53
- }): (original: any) => any;
54
- export declare function injectable(instanceCallback: InstanceCallback<any>): (original: any) => any;
55
- export declare function injectable(category: string | {
56
- new (...args: any[]): any;
57
- }, instanceCallback: InstanceCallback<any>): (original: any) => any;
64
+ export declare function injectable(cfg: Partial<InjectableConfig>): (original: any) => any;
65
+ /**
66
+ * @description Decorator that marks a class as available for dependency injection.
67
+ * @summary Defines a class as an injectable that can be retrieved from the registry.
68
+ * When applied to a class, replaces its constructor with one that returns an instance.
69
+ *
70
+ * @param category Defaults to the class category. Useful when minification occurs and names are changed,
71
+ * or when you want to upcast the object to a different type.
72
+ * @param {Partial<InjectableConfig>} cfg=DefaultInjectableConfig Allows overriding the default singleton behavior and adding a callback function.
73
+ *
74
+ * @return {function(any): any} A decorator function that transforms the class into an injectable.
75
+ *
76
+ * @function injectable
77
+ */
78
+ export declare function injectable(category: string | Constructor, cfg: Partial<InjectableConfig>): (original: any) => any;
79
+ /**
80
+ * @description Convenience decorator to register an injectable as a singleton.
81
+ * @summary Wraps {@link injectable} forcing the singleton lifecycle so only one instance is created and reused.
82
+ * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.
83
+ * @param {Omit<InjectableConfig, "singleton">} [cfg] Additional injectable configuration excluding the singleton flag.
84
+ * @return {function(any): any} A class decorator that registers the target as a singleton injectable.
85
+ * @function singleton
86
+ * @category Class Decorators
87
+ */
88
+ export declare function singleton(category?: string | Constructor, cfg?: Omit<InjectableConfig, "singleton">): (original: any) => any;
89
+ /**
90
+ * @description Convenience decorator to register an injectable as on-demand (non-singleton).
91
+ * @summary Wraps {@link injectable} forcing new instances to be created on every injection or retrieval.
92
+ * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.
93
+ * @param {Omit<InjectableConfig, "singleton">} [cfg] Additional injectable configuration excluding the singleton flag.
94
+ * @return {function(any): any} A class decorator that registers the target as a non-singleton injectable.
95
+ * @function onDemand
96
+ * @category Class Decorators
97
+ */
98
+ export declare function onDemand(category?: string | Constructor, cfg?: Omit<InjectableConfig, "singleton">): (original: any) => any;
58
99
  /**
59
100
  * @description Function type for transforming injectable instances before they're injected.
60
101
  * @summary Function which transforms a cached {@link injectable} instance before it's injected into a target object.
@@ -64,68 +105,65 @@ export declare function injectable(category: string | {
64
105
  * @return {any} The transformed injectable instance
65
106
  *
66
107
  * @typedef {Function} InstanceTransformer
108
+ * @category Decorators
67
109
  * @memberOf module:injectable-decorators
68
110
  */
69
111
  export type InstanceTransformer = (injectable: any, obj: any) => any;
112
+ /**
113
+ * @description Options for the property-level @inject decorator.
114
+ * @summary Allows specifying constructor arguments and an optional transformer to be applied to the resolved instance.
115
+ * @property {any[]} [args] Optional constructor arguments to use when building the injectable instance.
116
+ * @property {InstanceTransformer} [transformer] Optional function to transform the instance before assignment.
117
+ * @typedef InjectOptions
118
+ * @memberOf module:injectable-decorators
119
+ */
120
+ export type InjectOptions = {
121
+ args?: any[];
122
+ transformer?: InstanceTransformer;
123
+ };
124
+ /**
125
+ * @description Property decorator that injects a dependency into a class property.
126
+ * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
127
+ * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.
128
+ *
129
+ * @return {function(any, any): void} A property decorator function that sets up the dependency injection.
130
+ *
131
+ * @function inject
132
+ */
133
+ export declare function inject(): (target: any, propertyKey: any) => void;
70
134
  /**
71
135
  * @description Property decorator that injects a dependency into a class property.
72
136
  * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
73
137
  * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.
74
138
  *
75
- * Injected properties should be described like so:
76
- * <pre>
77
- * class ClassName {
78
- * ...
139
+ * @param {string} category Defaults to the class name derived from the property type. Useful when minification occurs
140
+ * and names are changed, or when you want to upcast the object to a different type.
141
+ * @return {function(any, any): void} A property decorator function that sets up the dependency injection.
79
142
  *
80
- * @inject()
81
- * propertyName!: InjectableClass;
143
+ * @function inject
144
+ */
145
+ export declare function inject(category: string | Constructor): (target: any, propertyKey: any) => void;
146
+ /**
147
+ * @description Property decorator that injects a dependency into a class property.
148
+ * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
149
+ * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.
82
150
  *
83
- * ...
84
- * }
85
- * </pre>
151
+ * @param {Partial<InjectOptions>} [cfg={}] Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand
152
+ * @return {function(any, any): void} A property decorator function that sets up the dependency injection.
86
153
  *
87
- * where InjectableClass is the class you want to inject.
88
- * Notice the use of '!:' to ensure the transpiler the property will be set outside the constructor but will always be defined.
89
- * For projects where minification occurs, you should use the category param to ensure the name is the same throughout.
154
+ * @function inject
155
+ */
156
+ export declare function inject(cfg: Partial<InjectOptions>): (target: any, propertyKey: any) => void;
157
+ /**
158
+ * @description Property decorator that injects a dependency into a class property.
159
+ * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
160
+ * The property must be typed for the requested dependency. Only concrete classes are supported; generics are not.
90
161
  *
91
- * @param {string} [category] Defaults to the class name derived from the property type. Useful when minification occurs
162
+ * @param {string} category Defaults to the class name derived from the property type. Useful when minification occurs
92
163
  * and names are changed, or when you want to upcast the object to a different type.
93
- * @param {InstanceTransformer} [transformer] Optional function to transform the injectable instance before it's injected.
94
- * @return {Function} A property decorator function that sets up the dependency injection.
164
+ * @param {Partial<InjectOptions>} cfg={} Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand
165
+ * @return {function(any, any): void} A property decorator function that sets up the dependency injection.
95
166
  *
96
167
  * @function inject
97
- * @category Property Decorators
98
- *
99
- * @mermaid
100
- * sequenceDiagram
101
- * participant Client
102
- * participant Decorator
103
- * participant Injectables
104
- *
105
- * Client->>Decorator: @inject()
106
- * Decorator->>Decorator: Get type from property
107
- * Decorator->>Decorator: Define metadata
108
- * Decorator->>Decorator: Define property getter
109
- *
110
- * Note over Decorator: When property accessed
111
- * Client->>Decorator: access property
112
- * Decorator->>Decorator: Check if instance exists
113
- * alt Instance exists in WeakMap
114
- * Decorator-->>Client: Return cached instance
115
- * else No instance
116
- * Decorator->>Injectables: get(name)
117
- * alt Injectable found
118
- * Injectables-->>Decorator: Return injectable instance
119
- * opt Has transformer
120
- * Decorator->>Decorator: Call transformer
121
- * end
122
- * Decorator->>Decorator: Store in WeakMap
123
- * Decorator-->>Client: Return instance
124
- * else No injectable
125
- * Decorator-->>Client: Throw error
126
- * end
127
- * end
128
168
  */
129
- export declare function inject(category?: symbol | string | {
130
- new (...args: any[]): any;
131
- }, transformer?: InstanceTransformer, ...args: any[]): (target: any, propertyKey?: any) => void;
169
+ export declare function inject(category: string | Constructor, cfg: Partial<InjectOptions>): (target: any, propertyKey: any) => void;
@@ -1,29 +1,63 @@
1
- import { InjectablesKeys } from "./constants.js";
1
+ import { DefaultInjectablesConfig, InjectablesKeys } from "./constants.js";
2
2
  import { Injectables } from "./Injectables.js";
3
- import { getTypeFromDecorator } from "./utils.js";
3
+ import { getInjectKey, getTypeFromDecorator } from "./utils.js";
4
4
  /**
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
9
- * @function getInjectKey
10
- * @memberOf module:injectable-decorators
5
+ * @description Decorator that marks a class as available for dependency injection.
6
+ * @summary Defines a class as an injectable that can be retrieved from the registry.
7
+ * When applied to a class, replaces its constructor with one that returns an instance.
8
+ *
9
+ * @param {string | Constructor} [category] Defaults to the class category. Useful when minification occurs and names are changed,
10
+ * or when you want to upcast the object to a different type.
11
+ * @param {Partial<InjectableConfig>} [cfg=DefaultInjectableConfig] Allows overriding the default singleton behavior and adding a callback function.
12
+ *
13
+ * @return {function(any): any} A decorator function that transforms the class into an injectable.
14
+ *
15
+ * @function injectable
16
+ * @category Class Decorators
17
+ *
18
+ * @mermaid
19
+ * sequenceDiagram
20
+ * participant Client
21
+ * participant Decorator
22
+ * participant Injectables
23
+ *
24
+ * Client->>Decorator: @injectable()
25
+ * Decorator->>Decorator: Create new constructor
26
+ *
27
+ * Note over Decorator: When new instance requested
28
+ * Decorator->>Injectables: get(category)
29
+ * alt Instance exists
30
+ * Injectables-->>Decorator: Return existing instance
31
+ * else No instance
32
+ * Decorator->>Injectables: register(original, category)
33
+ * Decorator->>Injectables: get(category)
34
+ * Injectables-->>Decorator: Return new instance
35
+ * opt Has callback
36
+ * Decorator->>Decorator: Call instanceCallback
37
+ * end
38
+ * end
39
+ * Decorator->>Decorator: Define metadata
40
+ * Decorator-->>Client: Return instance
11
41
  */
12
- export const getInjectKey = (key) => InjectablesKeys.REFLECT + key;
13
- export function injectable(name, cb) {
14
- const instanceCallback = (typeof name === "function" && !name.name ? name : cb);
15
- const category = typeof name === "string"
16
- ? name
17
- : cb
18
- ? name
19
- : typeof name === "function" && !name.name
20
- ? undefined
21
- : name;
42
+ export function injectable(category, cfg) {
43
+ cfg =
44
+ cfg ||
45
+ (typeof category === "object"
46
+ ? Object.assign(category, DefaultInjectablesConfig)
47
+ : DefaultInjectablesConfig);
48
+ category =
49
+ typeof category === "object"
50
+ ? undefined
51
+ : typeof category === "string"
52
+ ? category
53
+ : typeof category === "function" && category.name
54
+ ? category
55
+ : undefined;
22
56
  return (original) => {
23
57
  const symbol = Symbol.for(category || original.toString());
24
- const name = category || original.name;
58
+ category = category || original.name;
25
59
  const metadata = {
26
- class: name,
60
+ class: category,
27
61
  symbol: symbol,
28
62
  };
29
63
  Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECTABLE), metadata, original);
@@ -41,16 +75,36 @@ export function injectable(name, cb) {
41
75
  configurable: false,
42
76
  value: original.prototype.constructor.name,
43
77
  });
44
- const opts = {
45
- singleton: true,
46
- callback: instanceCallback,
47
- };
48
78
  Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECTABLE), metadata, newConstructor);
49
- Injectables.register(original, symbol, opts);
79
+ Injectables.register(original, symbol, cfg);
50
80
  // return new constructor (will override original)
51
81
  return newConstructor;
52
82
  };
53
83
  }
84
+ /**
85
+ * @description Convenience decorator to register an injectable as a singleton.
86
+ * @summary Wraps {@link injectable} forcing the singleton lifecycle so only one instance is created and reused.
87
+ * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.
88
+ * @param {Omit<InjectableConfig, "singleton">} [cfg] Additional injectable configuration excluding the singleton flag.
89
+ * @return {function(any): any} A class decorator that registers the target as a singleton injectable.
90
+ * @function singleton
91
+ * @category Class Decorators
92
+ */
93
+ export function singleton(category, cfg) {
94
+ return injectable(category, Object.assign({}, cfg || {}, { singleton: true }));
95
+ }
96
+ /**
97
+ * @description Convenience decorator to register an injectable as on-demand (non-singleton).
98
+ * @summary Wraps {@link injectable} forcing new instances to be created on every injection or retrieval.
99
+ * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.
100
+ * @param {Omit<InjectableConfig, "singleton">} [cfg] Additional injectable configuration excluding the singleton flag.
101
+ * @return {function(any): any} A class decorator that registers the target as a non-singleton injectable.
102
+ * @function onDemand
103
+ * @category Class Decorators
104
+ */
105
+ export function onDemand(category, cfg) {
106
+ return injectable(category, Object.assign({}, cfg || {}, { singleton: false }));
107
+ }
54
108
  /**
55
109
  * @description Property decorator that injects a dependency into a class property.
56
110
  * @summary Allows for the injection of an {@link injectable} decorated dependency into a class property.
@@ -74,8 +128,8 @@ export function injectable(name, cb) {
74
128
  *
75
129
  * @param {string} [category] Defaults to the class name derived from the property type. Useful when minification occurs
76
130
  * and names are changed, or when you want to upcast the object to a different type.
77
- * @param {InstanceTransformer} [transformer] Optional function to transform the injectable instance before it's injected.
78
- * @return {Function} A property decorator function that sets up the dependency injection.
131
+ * @param {Partial<InjectOptions>} [cfg={}] Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand
132
+ * @return {function(any, any): void} A property decorator function that sets up the dependency injection.
79
133
  *
80
134
  * @function inject
81
135
  * @category Property Decorators
@@ -110,39 +164,43 @@ export function injectable(name, cb) {
110
164
  * end
111
165
  * end
112
166
  */
113
- export function inject(category, transformer, ...args) {
167
+ export function inject(category, cfg) {
114
168
  return (target, propertyKey) => {
115
- const values = new WeakMap();
116
- const name = category || getTypeFromDecorator(target, propertyKey);
169
+ const config = (cfg || typeof category === "object" ? category : {});
170
+ const name = (typeof category !== "object" &&
171
+ category) ||
172
+ getTypeFromDecorator(target, propertyKey);
117
173
  if (!name) {
118
174
  throw new Error(`Could not get Type from decorator`);
119
175
  }
120
176
  Reflect.defineMetadata(getInjectKey(InjectablesKeys.INJECT), {
121
177
  injectable: name,
122
178
  }, target, propertyKey);
179
+ const values = new WeakMap();
123
180
  Object.defineProperty(target, propertyKey, {
124
181
  configurable: true,
125
182
  get() {
126
183
  const descriptor = Object.getOwnPropertyDescriptor(target, propertyKey);
127
184
  if (descriptor.configurable) {
185
+ // let /obj: any;
128
186
  Object.defineProperty(this, propertyKey, {
129
187
  enumerable: true,
130
188
  configurable: false,
131
189
  get() {
132
190
  let obj = values.get(this);
133
- if (!obj) {
134
- obj = Injectables.get(name, ...args);
135
- if (!obj)
136
- throw new Error(`Could not get Injectable ${name.toString()} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`);
137
- if (transformer)
138
- try {
139
- obj = transformer(obj, target);
140
- }
141
- catch (e) {
142
- console.error(e);
143
- }
144
- values.set(this, obj);
145
- }
191
+ if (obj)
192
+ return obj;
193
+ obj = Injectables.get(name, ...(config.args || []));
194
+ if (!obj)
195
+ throw new Error(`Could not get Injectable ${name.toString()} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`);
196
+ if (config.transformer)
197
+ try {
198
+ obj = config.transformer(obj, target);
199
+ }
200
+ catch (e) {
201
+ console.error(e);
202
+ }
203
+ values.set(this, obj);
146
204
  return obj;
147
205
  },
148
206
  });
@@ -152,4 +210,4 @@ export function inject(category, transformer, ...args) {
152
210
  });
153
211
  };
154
212
  }
155
- //# 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\";\nimport {\n  InjectableMetadata,\n  InjectableOptions,\n  InstanceCallback,\n} from \"./types\";\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 */\nexport const 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(): (original: any) => any;\nexport function injectable(\n  category: string | { new (...args: any[]): any }\n): (original: any) => any;\nexport function injectable(\n  instanceCallback: InstanceCallback<any>\n): (original: any) => any;\nexport function injectable(\n  category: string | { new (...args: any[]): any },\n  instanceCallback: InstanceCallback<any>\n): (original: any) => any;\nexport function injectable(\n  name?: string | { new (...args: any[]): any } | InstanceCallback<any>,\n  cb?: InstanceCallback<any>\n) {\n  const instanceCallback = (\n    typeof name === \"function\" && !name.name ? name : cb\n  ) as InstanceCallback<any> | undefined;\n  const category =\n    typeof name === \"string\"\n      ? name\n      : cb\n        ? name\n        : typeof name === \"function\" && !name.name\n          ? undefined\n          : name;\n\n  return (original: any) => {\n    const symbol = Symbol.for(category || original.toString());\n    const name = category || original.name;\n\n    const metadata: InjectableMetadata = {\n      class: name,\n      symbol: symbol,\n    };\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECTABLE),\n      metadata,\n      original\n    );\n    // the new constructor behaviour\n    const newConstructor: any = function (...args: any[]) {\n      return Injectables.get<any>(symbol, ...args);\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\n    const opts: InjectableOptions<any> = {\n      singleton: true,\n      callback: instanceCallback as InstanceCallback<any>,\n    };\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECTABLE),\n      metadata,\n      newConstructor\n    );\n\n    Injectables.register(original, symbol, opts);\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(\n  category?: symbol | string | { new (...args: any[]): any },\n  transformer?: InstanceTransformer,\n  ...args: any[]\n) {\n  return (target: any, propertyKey?: any) => {\n    const values = new WeakMap();\n\n    const name: symbol | string | { new (...args: any[]): any } | undefined =\n      category || getTypeFromDecorator(target, propertyKey);\n    if (!name) {\n      throw new Error(`Could not get Type from decorator`);\n    }\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, ...args);\n                if (!obj)\n                  throw new Error(\n                    `Could not get Injectable ${name.toString()} 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"]}
213
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import { DefaultInjectablesConfig, InjectablesKeys } from \"./constants\";\nimport { Injectables } from \"./Injectables\";\nimport { getInjectKey, getTypeFromDecorator } from \"./utils\";\nimport { InjectableMetadata, InstanceCallback } from \"./types\";\n\n/**\n * @description Configuration options for the @injectable decorator.\n * @summary Controls lifecycle (singleton vs on-demand) and an optional instance transformation callback executed post-construction.\n * @template T The instance type affected by the callback when provided.\n * @property {boolean} singleton When true, a single instance is shared (singleton). When false, instances are created on demand.\n * @property {InstanceCallback<T>} [callback] Optional hook to transform the instance after it is constructed.\n * @typedef InjectableConfig\n * @memberOf module:injectable-decorators\n */\nexport type InjectableConfig = {\n  singleton: boolean;\n  callback?: InstanceCallback<any>;\n};\n\n/**\n * @description Generic constructor type for class-like values.\n * @summary Represents any class that can be instantiated with arbitrary arguments, producing an instance of type T.\n * @template T The instance type created by the constructor.\n * @typedef Constructor\n * @example\n * // Using Constructor to type a factory\n * function make<T>(Ctor: Constructor<T>, ...args: any[]): T {\n *   return new Ctor(...args);\n * }\n * @memberOf module:injectable-decorators\n */\nexport type Constructor<T = any> = { new (...args: any[]): T };\n\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns an instance.\n *\n * @return {function(any): any} A decorator function that transforms the class into an injectable.\n *\n * @function injectable\n */\nexport function injectable(): (original: any) => any;\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns an instance.\n *\n * @param {string | Constructor} category Defaults to the class category. Useful when minification occurs and names are changed,\n * or when you want to upcast the object to a different type.\n *\n * @return {function(any): any} A decorator function that transforms the class into an injectable.\n *\n * @function injectable\n */\nexport function injectable(\n  category: string | Constructor\n): (original: any) => any;\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns an instance.\n *\n * @param {Partial<InjectableConfig>} cfg=DefaultInjectableConfig Allows overriding the default singleton behavior and adding a callback function.\n *\n * @return {function(any): any} A decorator function that transforms the class into an injectable.\n *\n * @function injectable\n */\nexport function injectable(\n  cfg: Partial<InjectableConfig>\n): (original: any) => any;\n\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns an instance.\n *\n * @param category Defaults to the class category. Useful when minification occurs and names are changed,\n * or when you want to upcast the object to a different type.\n * @param {Partial<InjectableConfig>} cfg=DefaultInjectableConfig Allows overriding the default singleton behavior and adding a callback function.\n *\n * @return {function(any): any} A decorator function that transforms the class into an injectable.\n *\n * @function injectable\n */\nexport function injectable(\n  category: string | Constructor,\n  cfg: Partial<InjectableConfig>\n): (original: any) => any;\n/**\n * @description Decorator that marks a class as available for dependency injection.\n * @summary Defines a class as an injectable that can be retrieved from the registry.\n * When applied to a class, replaces its constructor with one that returns an instance.\n *\n * @param {string | Constructor} [category] Defaults to the class category. Useful when minification occurs and names are changed,\n * or when you want to upcast the object to a different type.\n * @param {Partial<InjectableConfig>} [cfg=DefaultInjectableConfig] Allows overriding the default singleton behavior and adding a callback function.\n *\n * @return {function(any): any} 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(category)\n *   alt Instance exists\n *     Injectables-->>Decorator: Return existing instance\n *   else No instance\n *     Decorator->>Injectables: register(original, category)\n *     Decorator->>Injectables: get(category)\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 | Constructor | Partial<InjectableConfig>,\n  cfg?: Partial<InjectableConfig>\n) {\n  cfg =\n    cfg ||\n    (typeof category === \"object\"\n      ? Object.assign(category as Record<any, any>, DefaultInjectablesConfig)\n      : DefaultInjectablesConfig);\n  category =\n    typeof category === \"object\"\n      ? undefined\n      : typeof category === \"string\"\n        ? category\n        : typeof category === \"function\" && category.name\n          ? category\n          : undefined;\n\n  return (original: any) => {\n    const symbol = Symbol.for(category || original.toString());\n    category = category || original.name;\n\n    const metadata: InjectableMetadata = {\n      class: category as string,\n      symbol: symbol,\n    };\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECTABLE),\n      metadata,\n      original\n    );\n    // the new constructor behaviour\n    const newConstructor: any = function (...args: any[]) {\n      return Injectables.get<any>(symbol, ...args);\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\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECTABLE),\n      metadata,\n      newConstructor\n    );\n\n    Injectables.register(original, symbol, cfg);\n    // return new constructor (will override original)\n    return newConstructor;\n  };\n}\n\n/**\n * @description Convenience decorator to register an injectable as a singleton.\n * @summary Wraps {@link injectable} forcing the singleton lifecycle so only one instance is created and reused.\n * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.\n * @param {Omit<InjectableConfig, \"singleton\">} [cfg] Additional injectable configuration excluding the singleton flag.\n * @return {function(any): any} A class decorator that registers the target as a singleton injectable.\n * @function singleton\n * @category Class Decorators\n */\nexport function singleton(\n  category?: string | Constructor,\n  cfg?: Omit<InjectableConfig, \"singleton\">\n) {\n  return injectable(\n    category as any,\n    Object.assign({}, cfg || {}, { singleton: true })\n  );\n}\n\n/**\n * @description Convenience decorator to register an injectable as on-demand (non-singleton).\n * @summary Wraps {@link injectable} forcing new instances to be created on every injection or retrieval.\n * @param {string|Constructor} [category] Optional explicit category/symbol source; defaults to the class.\n * @param {Omit<InjectableConfig, \"singleton\">} [cfg] Additional injectable configuration excluding the singleton flag.\n * @return {function(any): any} A class decorator that registers the target as a non-singleton injectable.\n * @function onDemand\n * @category Class Decorators\n */\nexport function onDemand(\n  category?: string | Constructor,\n  cfg?: Omit<InjectableConfig, \"singleton\">\n) {\n  return injectable(\n    category as any,\n    Object.assign({}, cfg || {}, { singleton: false })\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 * @category Decorators\n * @memberOf module:injectable-decorators\n */\nexport type InstanceTransformer = (injectable: any, obj: any) => any;\n\n/**\n * @description Options for the property-level @inject decorator.\n * @summary Allows specifying constructor arguments and an optional transformer to be applied to the resolved instance.\n * @property {any[]} [args] Optional constructor arguments to use when building the injectable instance.\n * @property {InstanceTransformer} [transformer] Optional function to transform the instance before assignment.\n * @typedef InjectOptions\n * @memberOf module:injectable-decorators\n */\nexport type InjectOptions = {\n  args?: any[];\n  transformer?: InstanceTransformer;\n};\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 * @return {function(any, any): void} A property decorator function that sets up the dependency injection.\n *\n * @function inject\n */\nexport function inject(): (target: any, propertyKey: any) => void;\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 * @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 * @return {function(any, any): void} A property decorator function that sets up the dependency injection.\n *\n * @function inject\n */\nexport function inject(\n  category: string | Constructor\n): (target: any, propertyKey: any) => void;\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 * @param {Partial<InjectOptions>} [cfg={}] Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand\n * @return {function(any, any): void} A property decorator function that sets up the dependency injection.\n *\n * @function inject\n */\nexport function inject(\n  cfg: Partial<InjectOptions>\n): (target: any, propertyKey: any) => void;\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 * @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 {Partial<InjectOptions>} cfg={} Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand\n * @return {function(any, any): void} A property decorator function that sets up the dependency injection.\n *\n * @function inject\n */\nexport function inject(\n  category: string | Constructor,\n  cfg: Partial<InjectOptions>\n): (target: any, propertyKey: any) => void;\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 {Partial<InjectOptions>} [cfg={}] Optional function to transform the injectable instance before it's injected, or arguments to pass the constructor when injecting onDemand\n * @return {function(any, any): void} 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(\n  category?: symbol | string | Constructor | Partial<InjectOptions>,\n  cfg?: Partial<InjectOptions>\n) {\n  return (target: any, propertyKey: any) => {\n    const config: InjectOptions = (\n      cfg || typeof category === \"object\" ? category : {}\n    ) as InjectOptions;\n\n    const name: symbol | string | Constructor | undefined =\n      (typeof category !== \"object\" &&\n        (category as symbol | string | Constructor)) ||\n      getTypeFromDecorator(target, propertyKey);\n    if (!name) {\n      throw new Error(`Could not get Type from decorator`);\n    }\n\n    Reflect.defineMetadata(\n      getInjectKey(InjectablesKeys.INJECT),\n      {\n        injectable: name,\n      },\n      target,\n      propertyKey\n    );\n\n    const values = new WeakMap();\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          // let /obj: any;\n          Object.defineProperty(this, propertyKey, {\n            enumerable: true,\n            configurable: false,\n            get(this: any) {\n              let obj = values.get(this);\n              if (obj) return obj;\n              obj = Injectables.get(name, ...(config.args || []));\n              if (!obj)\n                throw new Error(\n                  `Could not get Injectable ${name.toString()} to inject in ${target.constructor ? target.constructor.name : target.name}'s ${propertyKey}`\n                );\n              if (config.transformer)\n                try {\n                  obj = config.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"]}
@@ -18,4 +18,4 @@ export * from "./utils";
18
18
  * @const VERSION
19
19
  * @memberOf module:injectable-decorators
20
20
  */
21
- export declare const VERSION = "1.6.7";
21
+ export declare const VERSION = "1.6.9";
package/lib/esm/index.js CHANGED
@@ -18,5 +18,5 @@ export * from "./utils.js";
18
18
  * @const VERSION
19
19
  * @memberOf module:injectable-decorators
20
20
  */
21
- export const VERSION = "1.6.7";
21
+ export const VERSION = "1.6.9";
22
22
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7R0FPRztBQUVILCtCQUE0QjtBQUM1QixnQ0FBNkI7QUFDN0IsaUNBQThCO0FBQzlCLDhCQUEyQjtBQUMzQiwyQkFBd0I7QUFDeEIsMkJBQXdCO0FBRXhCOzs7OztHQUtHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLGFBQWEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGRlc2NyaXB0aW9uIEEgbGlnaHR3ZWlnaHQgZGVwZW5kZW5jeSBpbmplY3Rpb24gbGlicmFyeSBmb3IgVHlwZVNjcmlwdCBhcHBsaWNhdGlvbnMuXG4gKiBAc3VtbWFyeSBBZGRzIGEgc2ltcGxlIEluamVjdGFibGVzIGltcGxlbWVudGF0aW9uIHRvIGNyZWF0ZSBzaW5nbGV0b24gaW5zdGFuY2VzIG9mIGFuIG9iamVjdFxuICogYW5kIGVhc2lseSBpbmplY3QgaXQgaW50byBvdGhlciBvYmplY3RzLiBQcm92aWRlcyBkZWNvcmF0b3JzIGZvciBtYXJraW5nIGNsYXNzZXMgYXMgaW5qZWN0YWJsZVxuICogYW5kIGZvciBpbmplY3RpbmcgZGVwZW5kZW5jaWVzIGludG8gY2xhc3MgcHJvcGVydGllcy5cbiAqXG4gKiBAbW9kdWxlIGluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL2NvbnN0YW50c1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vSW5qZWN0YWJsZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3JlZ2lzdHJ5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQ3VycmVudCB2ZXJzaW9uIG9mIHRoZSBpbmplY3RhYmxlLWRlY29yYXRvcnMgbGlicmFyeS5cbiAqIEBzdW1tYXJ5IERlZmluZWQgb24gbGlicmFyeSBidWlsZC4gSG9sZHMgdGhlIGxpYnJhcnkncyBjdXJyZW50IHZlcnNpb24gc3RyaW5nLlxuICogQGNvbnN0IFZFUlNJT05cbiAqIEBtZW1iZXJPZiBtb2R1bGU6aW5qZWN0YWJsZS1kZWNvcmF0b3JzXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gXCIjI1ZFUlNJT04jI1wiO1xuIl19