@decaf-ts/injectable-decorators 1.6.3 → 1.6.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/injectable-decorators.cjs +70 -44
- package/dist/injectable-decorators.esm.cjs +70 -45
- package/lib/Injectables.cjs +4 -4
- package/lib/Injectables.d.ts +5 -3
- package/lib/decorators.cjs +27 -17
- package/lib/decorators.d.ts +14 -2
- package/lib/esm/Injectables.d.ts +5 -3
- package/lib/esm/Injectables.js +4 -4
- package/lib/esm/decorators.d.ts +14 -2
- package/lib/esm/decorators.js +25 -17
- package/lib/esm/index.d.ts +2 -1
- package/lib/esm/index.js +3 -2
- package/lib/esm/registry.d.ts +17 -10
- package/lib/esm/registry.js +36 -24
- package/lib/esm/types.d.ts +16 -0
- package/lib/esm/types.js +2 -0
- package/lib/esm/utils.d.ts +1 -1
- package/lib/esm/utils.js +11 -2
- package/lib/index.cjs +3 -2
- package/lib/index.d.ts +2 -1
- package/lib/registry.cjs +36 -24
- package/lib/registry.d.ts +17 -10
- package/lib/types.cjs +3 -0
- package/lib/types.d.ts +16 -0
- package/lib/utils.cjs +11 -2
- package/lib/utils.d.ts +1 -1
- package/package.json +1 -1
package/lib/index.cjs
CHANGED
|
@@ -27,6 +27,7 @@ __exportStar(require("./constants.cjs"), exports);
|
|
|
27
27
|
__exportStar(require("./decorators.cjs"), exports);
|
|
28
28
|
__exportStar(require("./Injectables.cjs"), exports);
|
|
29
29
|
__exportStar(require("./registry.cjs"), exports);
|
|
30
|
+
__exportStar(require("./types.cjs"), exports);
|
|
30
31
|
__exportStar(require("./utils.cjs"), exports);
|
|
31
32
|
/**
|
|
32
33
|
* @description Current version of the injectable-decorators library.
|
|
@@ -34,5 +35,5 @@ __exportStar(require("./utils.cjs"), exports);
|
|
|
34
35
|
* @const VERSION
|
|
35
36
|
* @memberOf module:injectable-decorators
|
|
36
37
|
*/
|
|
37
|
-
exports.VERSION = "1.6.
|
|
38
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
38
|
+
exports.VERSION = "1.6.4";
|
|
39
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7O0dBT0c7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsa0RBQTRCO0FBQzVCLG1EQUE2QjtBQUM3QixvREFBOEI7QUFDOUIsaURBQTJCO0FBQzNCLDhDQUF3QjtBQUN4Qiw4Q0FBd0I7QUFFeEI7Ozs7O0dBS0c7QUFDVSxRQUFBLE9BQU8sR0FBRyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBkZXNjcmlwdGlvbiBBIGxpZ2h0d2VpZ2h0IGRlcGVuZGVuY3kgaW5qZWN0aW9uIGxpYnJhcnkgZm9yIFR5cGVTY3JpcHQgYXBwbGljYXRpb25zLlxuICogQHN1bW1hcnkgQWRkcyBhIHNpbXBsZSBJbmplY3RhYmxlcyBpbXBsZW1lbnRhdGlvbiB0byBjcmVhdGUgc2luZ2xldG9uIGluc3RhbmNlcyBvZiBhbiBvYmplY3RcbiAqIGFuZCBlYXNpbHkgaW5qZWN0IGl0IGludG8gb3RoZXIgb2JqZWN0cy4gUHJvdmlkZXMgZGVjb3JhdG9ycyBmb3IgbWFya2luZyBjbGFzc2VzIGFzIGluamVjdGFibGVcbiAqIGFuZCBmb3IgaW5qZWN0aW5nIGRlcGVuZGVuY2llcyBpbnRvIGNsYXNzIHByb3BlcnRpZXMuXG4gKlxuICogQG1vZHVsZSBpbmplY3RhYmxlLWRlY29yYXRvcnNcbiAqL1xuXG5leHBvcnQgKiBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL0luamVjdGFibGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9yZWdpc3RyeVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgaW5qZWN0YWJsZS1kZWNvcmF0b3JzIGxpYnJhcnkuXG4gKiBAc3VtbWFyeSBEZWZpbmVkIG9uIGxpYnJhcnkgYnVpbGQuIEhvbGRzIHRoZSBsaWJyYXJ5J3MgY3VycmVudCB2ZXJzaW9uIHN0cmluZy5cbiAqIEBjb25zdCBWRVJTSU9OXG4gKiBAbWVtYmVyT2YgbW9kdWxlOmluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbiJdfQ==
|
package/lib/index.d.ts
CHANGED
|
@@ -10,6 +10,7 @@ export * from "./constants";
|
|
|
10
10
|
export * from "./decorators";
|
|
11
11
|
export * from "./Injectables";
|
|
12
12
|
export * from "./registry";
|
|
13
|
+
export * from "./types";
|
|
13
14
|
export * from "./utils";
|
|
14
15
|
/**
|
|
15
16
|
* @description Current version of the injectable-decorators library.
|
|
@@ -17,4 +18,4 @@ export * from "./utils";
|
|
|
17
18
|
* @const VERSION
|
|
18
19
|
* @memberOf module:injectable-decorators
|
|
19
20
|
*/
|
|
20
|
-
export declare const VERSION = "1.6.
|
|
21
|
+
export declare const VERSION = "1.6.4";
|
package/lib/registry.cjs
CHANGED
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.InjectableRegistryImp = void 0;
|
|
4
|
+
const constants_1 = require("./constants.cjs");
|
|
5
|
+
const decorators_1 = require("./decorators.cjs");
|
|
4
6
|
/**
|
|
5
7
|
* @description Default implementation of the InjectablesRegistry interface.
|
|
6
8
|
* @summary Holds the various {@link Injectable}s in a cache and provides methods to register, retrieve, and build them.
|
|
@@ -47,54 +49,64 @@ class InjectableRegistryImp {
|
|
|
47
49
|
constructor() {
|
|
48
50
|
this.cache = {};
|
|
49
51
|
}
|
|
52
|
+
has(name) {
|
|
53
|
+
if (typeof name === "symbol")
|
|
54
|
+
return name in this.cache;
|
|
55
|
+
return Symbol.for(name.toString()) in this.cache;
|
|
56
|
+
}
|
|
50
57
|
/**
|
|
51
58
|
* @inheritDoc
|
|
52
59
|
*/
|
|
53
60
|
get(name, ...args) {
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
return innerCache.instance || this.build(buildDef, ...args);
|
|
60
|
-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
61
|
+
if (typeof name === "string")
|
|
62
|
+
name = Symbol.for(name);
|
|
63
|
+
if (typeof name !== "symbol") {
|
|
64
|
+
const meta = Reflect.getMetadata((0, decorators_1.getInjectKey)(constants_1.InjectablesKeys.INJECTABLE), name);
|
|
65
|
+
name = meta?.symbol || Symbol.for(name.toString());
|
|
61
66
|
}
|
|
62
|
-
|
|
67
|
+
if (!name)
|
|
68
|
+
throw new Error(`Injectable ${name} not found`);
|
|
69
|
+
if (!(name in this.cache)) {
|
|
63
70
|
return undefined;
|
|
64
71
|
}
|
|
72
|
+
const cache = this.cache[name];
|
|
73
|
+
if (!cache.options.singleton && !cache.instance)
|
|
74
|
+
return this.build(name, ...args);
|
|
75
|
+
return cache.instance || this.build(name, ...args);
|
|
65
76
|
}
|
|
66
77
|
/**
|
|
67
78
|
* @inheritDoc
|
|
68
79
|
*/
|
|
69
|
-
register(obj, category
|
|
80
|
+
register(obj, category, options, force = false) {
|
|
70
81
|
const castObj = obj;
|
|
71
82
|
const constructor = !castObj.name && castObj.constructor;
|
|
72
83
|
if (typeof castObj !== "function" && !constructor)
|
|
73
84
|
throw new Error(`Injectable registering failed. Missing Class name or constructor`);
|
|
74
|
-
const name = category ||
|
|
75
|
-
(constructor && constructor.name && constructor.name !== "Function"
|
|
76
|
-
? constructor.name
|
|
77
|
-
: castObj.name);
|
|
85
|
+
const name = category || Symbol.for(obj.toString());
|
|
78
86
|
if (!this.cache[name] || force)
|
|
79
87
|
this.cache[name] = {
|
|
80
88
|
instance: constructor ? obj : undefined,
|
|
81
|
-
constructor: !constructor ? obj :
|
|
82
|
-
|
|
89
|
+
constructor: !constructor ? obj : obj.constructor,
|
|
90
|
+
options: options,
|
|
83
91
|
};
|
|
84
92
|
}
|
|
85
93
|
/**
|
|
86
94
|
* @inheritDoc
|
|
87
95
|
*/
|
|
88
|
-
build(
|
|
89
|
-
const { constructor,
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
instance
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
+
build(name, ...args) {
|
|
97
|
+
const { constructor, options } = this.cache[name];
|
|
98
|
+
let instance;
|
|
99
|
+
try {
|
|
100
|
+
instance = new constructor(...args);
|
|
101
|
+
}
|
|
102
|
+
catch (e) {
|
|
103
|
+
throw new Error(`failed to build ${name.toString()} with args ${args}: ${e}`);
|
|
104
|
+
}
|
|
105
|
+
this.cache[name].instance = instance;
|
|
106
|
+
if (options.callback)
|
|
107
|
+
instance = options.callback(instance, ...args);
|
|
96
108
|
return instance;
|
|
97
109
|
}
|
|
98
110
|
}
|
|
99
111
|
exports.InjectableRegistryImp = InjectableRegistryImp;
|
|
100
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../src/registry.ts"],"names":[],"mappings":";;;AAmDA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAyCG;AACH,MAAa,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;AA7DD,sDA6DC","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"]}
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../src/registry.ts"],"names":[],"mappings":"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","sourcesContent":["import { InjectableDef, InjectableOptions } from \"./types\";\nimport { InjectablesKeys } from \"./constants\";\nimport { getInjectKey } from \"./decorators\";\n\n/**\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 {symbol} 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>(\n    name: symbol | string | { new (...args: any[]): T },\n    ...args: any[]\n  ): 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 options\n   * @param {any[]} args Additional arguments for registration (category, singleton flag, etc.)\n   * @return {void}\n   * @memberOf module:injectable-decorators\n   */\n  register<T>(\n    constructor: Injectable<T>,\n    category: symbol | undefined,\n    options: InjectableOptions<T>,\n    ...args: any[]\n  ): 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 {symbol} name 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>(name: symbol, ...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: Record<symbol, InjectableDef> = {};\n\n  has<T>(name: symbol | { new (...args: any[]): T }): boolean {\n    if (typeof name === \"symbol\") return name in this.cache;\n    return Symbol.for(name.toString()) in this.cache;\n  }\n\n  /**\n   * @inheritDoc\n   */\n  get<T>(\n    name: symbol | string | { new (...args: any[]): T },\n    ...args: any[]\n  ): T | undefined {\n    if (typeof name === \"string\") name = Symbol.for(name);\n    if (typeof name !== \"symbol\") {\n      const meta = Reflect.getMetadata(\n        getInjectKey(InjectablesKeys.INJECTABLE),\n        name\n      );\n      name = (meta?.symbol as symbol) || Symbol.for(name.toString());\n    }\n    if (!name) throw new Error(`Injectable ${name} not found`);\n\n    if (!((name as symbol) in this.cache)) {\n      return undefined;\n    }\n    const cache = this.cache[name];\n    if (!cache.options.singleton && !cache.instance)\n      return this.build<T>(name, ...args);\n    return cache.instance || this.build<T>(name, ...args);\n  }\n  /**\n   * @inheritDoc\n   */\n  register<T>(\n    obj: Injectable<T>,\n    category: symbol | undefined,\n    options: InjectableOptions<T>,\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 = category || Symbol.for((obj as any).toString());\n\n    if (!this.cache[name] || force)\n      this.cache[name] = {\n        instance: constructor ? obj : undefined,\n        constructor: !constructor ? obj : (obj as any).constructor,\n        options: options,\n      };\n  }\n  /**\n   * @inheritDoc\n   */\n  build<T>(name: symbol, ...args: any[]): T {\n    const { constructor, options } = this.cache[name];\n    let instance: T;\n    try {\n      instance = new constructor(...args);\n    } catch (e: unknown) {\n      throw new Error(\n        `failed to build ${name.toString()} with args ${args}: ${e}`\n      );\n    }\n    this.cache[name].instance = instance;\n    if (options.callback) instance = options.callback(instance, ...args);\n    return instance;\n  }\n}\n"]}
|
package/lib/registry.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { InjectableOptions } from "./types";
|
|
1
2
|
/**
|
|
2
3
|
* @description Type representing either a class constructor or an instance.
|
|
3
4
|
* @summary Defines an Injectable type that can be either a class constructor or an instance of a class.
|
|
@@ -20,32 +21,35 @@ export interface InjectablesRegistry {
|
|
|
20
21
|
* @description Fetches an injectable instance by its registered name.
|
|
21
22
|
* @summary Retrieves an {@link Injectable} from the registry by name, optionally passing constructor arguments.
|
|
22
23
|
* @template T Type of the injectable object to retrieve
|
|
23
|
-
* @param {
|
|
24
|
+
* @param {symbol} name The registered name of the injectable to retrieve
|
|
24
25
|
* @param {any[]} args Constructor arguments to pass when instantiating the injectable
|
|
25
26
|
* @return {Injectable<T> | undefined} The injectable instance or undefined if not found
|
|
26
27
|
* @memberOf module:injectable-decorators
|
|
27
28
|
*/
|
|
28
|
-
get<T>(name:
|
|
29
|
+
get<T>(name: symbol | string | {
|
|
30
|
+
new (...args: any[]): T;
|
|
31
|
+
}, ...args: any[]): T | undefined;
|
|
29
32
|
/**
|
|
30
33
|
* @description Adds a class or object to the injectable registry.
|
|
31
34
|
* @summary Registers an injectable constructor or instance with the registry, making it available for injection.
|
|
32
35
|
* @template T Type of the injectable object to register
|
|
33
36
|
* @param {Injectable<T>} constructor The class constructor or object instance to register
|
|
37
|
+
* @param options
|
|
34
38
|
* @param {any[]} args Additional arguments for registration (category, singleton flag, etc.)
|
|
35
39
|
* @return {void}
|
|
36
40
|
* @memberOf module:injectable-decorators
|
|
37
41
|
*/
|
|
38
|
-
register<T>(constructor: Injectable<T>, ...args: any[]): void;
|
|
42
|
+
register<T>(constructor: Injectable<T>, category: symbol | undefined, options: InjectableOptions<T>, ...args: any[]): void;
|
|
39
43
|
/**
|
|
40
44
|
* @description Creates a new instance of an injectable class.
|
|
41
45
|
* @summary Instantiates an injectable class using its constructor and the provided arguments.
|
|
42
46
|
* @template T Type of the object to build
|
|
43
|
-
* @param {
|
|
47
|
+
* @param {symbol} name Object containing the name of the injectable to build
|
|
44
48
|
* @param {any[]} args Constructor arguments to pass when instantiating the injectable
|
|
45
49
|
* @return {T} The newly created instance
|
|
46
50
|
* @memberOf module:injectable-decorators
|
|
47
51
|
*/
|
|
48
|
-
build<T>(
|
|
52
|
+
build<T>(name: symbol, ...args: any[]): T;
|
|
49
53
|
}
|
|
50
54
|
/**
|
|
51
55
|
* @description Default implementation of the InjectablesRegistry interface.
|
|
@@ -91,18 +95,21 @@ export interface InjectablesRegistry {
|
|
|
91
95
|
*/
|
|
92
96
|
export declare class InjectableRegistryImp implements InjectablesRegistry {
|
|
93
97
|
private cache;
|
|
98
|
+
has<T>(name: symbol | {
|
|
99
|
+
new (...args: any[]): T;
|
|
100
|
+
}): boolean;
|
|
94
101
|
/**
|
|
95
102
|
* @inheritDoc
|
|
96
103
|
*/
|
|
97
|
-
get<T>(name:
|
|
104
|
+
get<T>(name: symbol | string | {
|
|
105
|
+
new (...args: any[]): T;
|
|
106
|
+
}, ...args: any[]): T | undefined;
|
|
98
107
|
/**
|
|
99
108
|
* @inheritDoc
|
|
100
109
|
*/
|
|
101
|
-
register<T>(obj: Injectable<T>, category
|
|
110
|
+
register<T>(obj: Injectable<T>, category: symbol | undefined, options: InjectableOptions<T>, force?: boolean): void;
|
|
102
111
|
/**
|
|
103
112
|
* @inheritDoc
|
|
104
113
|
*/
|
|
105
|
-
build<T>(
|
|
106
|
-
name: string;
|
|
107
|
-
}, ...args: any[]): T;
|
|
114
|
+
build<T>(name: symbol, ...args: any[]): T;
|
|
108
115
|
}
|
package/lib/types.cjs
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEluc3RhbmNlQ2FsbGJhY2s8VD4gPSAoaW5zdGFuY2U6IFQsIC4uLmFyZ3M6IGFueVtdKSA9PiBUO1xuXG5leHBvcnQgdHlwZSBJbmplY3RhYmxlT3B0aW9uczxUPiA9IHtcbiAgc2luZ2xldG9uOiBib29sZWFuO1xuICBjYWxsYmFjazogSW5zdGFuY2VDYWxsYmFjazxUPjtcbn07XG5cbmV4cG9ydCB0eXBlIEluamVjdGFibGVEZWY8XG4gIFQgPSBhbnksXG4gIE9QVFMgZXh0ZW5kcyBJbmplY3RhYmxlT3B0aW9uczxUPiA9IEluamVjdGFibGVPcHRpb25zPFQ+LFxuPiA9IHtcbiAgb3B0aW9uczogT1BUUztcbiAgaW5zdGFuY2U/OiBhbnk7XG4gIGNvbnN0cnVjdG9yOiB7IG5ldyAoLi4uYXJnczogYW55W10pOiBhbnkgfTtcbn07XG5cbmV4cG9ydCB0eXBlIEluamVjdGFibGVNZXRhZGF0YSA9IHtcbiAgY2xhc3M6IHN0cmluZztcbiAgc3ltYm9sOiBzeW1ib2w7XG59O1xuIl19
|
package/lib/types.d.ts
ADDED
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export type InstanceCallback<T> = (instance: T, ...args: any[]) => T;
|
|
2
|
+
export type InjectableOptions<T> = {
|
|
3
|
+
singleton: boolean;
|
|
4
|
+
callback: InstanceCallback<T>;
|
|
5
|
+
};
|
|
6
|
+
export type InjectableDef<T = any, OPTS extends InjectableOptions<T> = InjectableOptions<T>> = {
|
|
7
|
+
options: OPTS;
|
|
8
|
+
instance?: any;
|
|
9
|
+
constructor: {
|
|
10
|
+
new (...args: any[]): any;
|
|
11
|
+
};
|
|
12
|
+
};
|
|
13
|
+
export type InjectableMetadata = {
|
|
14
|
+
class: string;
|
|
15
|
+
symbol: symbol;
|
|
16
|
+
};
|
package/lib/utils.cjs
CHANGED
|
@@ -3,6 +3,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.TypeKey = void 0;
|
|
4
4
|
exports.getTypeFromDecorator = getTypeFromDecorator;
|
|
5
5
|
require("reflect-metadata");
|
|
6
|
+
const decorators_1 = require("./decorators.cjs");
|
|
7
|
+
const constants_1 = require("./constants.cjs");
|
|
6
8
|
/**
|
|
7
9
|
* @description Reflection metadata key for accessing TypeScript type information.
|
|
8
10
|
* @summary Holds the key for retrieving the design type from TypeScript's reflection metadata.
|
|
@@ -21,6 +23,13 @@ exports.TypeKey = "design:type";
|
|
|
21
23
|
*/
|
|
22
24
|
function getTypeFromDecorator(model, propKey) {
|
|
23
25
|
const typeDef = Reflect.getMetadata(exports.TypeKey, model, propKey);
|
|
24
|
-
|
|
26
|
+
if (typeDef.name === "Function") {
|
|
27
|
+
return undefined;
|
|
28
|
+
}
|
|
29
|
+
const meta = Reflect.getMetadata((0, decorators_1.getInjectKey)(constants_1.InjectablesKeys.INJECTABLE), typeDef);
|
|
30
|
+
if (!meta) {
|
|
31
|
+
return undefined;
|
|
32
|
+
}
|
|
33
|
+
return meta.symbol;
|
|
25
34
|
}
|
|
26
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBcUJBLG9EQWdCQztBQXJDRCw0QkFBMEI7QUFDMUIsaURBQTRDO0FBQzVDLCtDQUE4QztBQUU5Qzs7Ozs7R0FLRztBQUNVLFFBQUEsT0FBTyxHQUFHLGFBQWEsQ0FBQztBQUVyQzs7Ozs7Ozs7R0FRRztBQUNILFNBQWdCLG9CQUFvQixDQUNsQyxLQUFVLEVBQ1YsT0FBd0I7SUFFeEIsTUFBTSxPQUFPLEdBQUcsT0FBTyxDQUFDLFdBQVcsQ0FBQyxlQUFPLEVBQUUsS0FBSyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0lBQzdELElBQUksT0FBTyxDQUFDLElBQUksS0FBSyxVQUFVLEVBQUUsQ0FBQztRQUNoQyxPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO0lBQ0QsTUFBTSxJQUFJLEdBQUcsT0FBTyxDQUFDLFdBQVcsQ0FDOUIsSUFBQSx5QkFBWSxFQUFDLDJCQUFlLENBQUMsVUFBVSxDQUFDLEVBQ3hDLE9BQU8sQ0FDUixDQUFDO0lBQ0YsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ1YsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUNELE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQztBQUNyQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFwicmVmbGVjdC1tZXRhZGF0YVwiO1xuaW1wb3J0IHsgZ2V0SW5qZWN0S2V5IH0gZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgSW5qZWN0YWJsZXNLZXlzIH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIFJlZmxlY3Rpb24gbWV0YWRhdGEga2V5IGZvciBhY2Nlc3NpbmcgVHlwZVNjcmlwdCB0eXBlIGluZm9ybWF0aW9uLlxuICogQHN1bW1hcnkgSG9sZHMgdGhlIGtleSBmb3IgcmV0cmlldmluZyB0aGUgZGVzaWduIHR5cGUgZnJvbSBUeXBlU2NyaXB0J3MgcmVmbGVjdGlvbiBtZXRhZGF0YS5cbiAqIEBjb25zdCBUeXBlS2V5XG4gKiBAbWVtYmVyT2YgbW9kdWxlOmluamVjdGFibGUtZGVjb3JhdG9yc1xuICovXG5leHBvcnQgY29uc3QgVHlwZUtleSA9IFwiZGVzaWduOnR5cGVcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRXh0cmFjdHMgdGhlIHR5cGUgbmFtZSBmcm9tIGEgZGVjb3JhdGVkIHByb3BlcnR5IHVzaW5nIHJlZmxlY3Rpb24uXG4gKiBAc3VtbWFyeSBSZXRyaWV2ZXMgdGhlIHR5cGUgZnJvbSBhIHByb3BlcnR5IGRlY29yYXRvciBieSBhY2Nlc3NpbmcgVHlwZVNjcmlwdCdzIHJlZmxlY3Rpb24gbWV0YWRhdGEuXG4gKiBAcGFyYW0ge2FueX0gbW9kZWwgVGhlIHRhcmdldCBvYmplY3QgY29udGFpbmluZyB0aGUgZGVjb3JhdGVkIHByb3BlcnR5XG4gKiBAcGFyYW0ge3N0cmluZyB8IHN5bWJvbH0gcHJvcEtleSBUaGUgcHJvcGVydHkga2V5IChuYW1lIG9yIHN5bWJvbCkgb2YgdGhlIGRlY29yYXRlZCBwcm9wZXJ0eVxuICogQHJldHVybiB7c3RyaW5nIHwgdW5kZWZpbmVkfSBUaGUgbmFtZSBvZiB0aGUgcHJvcGVydHkgdHlwZSwgb3IgdW5kZWZpbmVkIGlmIGl0J3MgYSBGdW5jdGlvbiB0eXBlXG4gKiBAZnVuY3Rpb24gZ2V0VHlwZUZyb21EZWNvcmF0b3JcbiAqIEBtZW1iZXJPZiBtb2R1bGU6aW5qZWN0YWJsZS1kZWNvcmF0b3JzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRUeXBlRnJvbURlY29yYXRvcihcbiAgbW9kZWw6IGFueSxcbiAgcHJvcEtleTogc3RyaW5nIHwgc3ltYm9sXG4pOiBzeW1ib2wgfCB1bmRlZmluZWQge1xuICBjb25zdCB0eXBlRGVmID0gUmVmbGVjdC5nZXRNZXRhZGF0YShUeXBlS2V5LCBtb2RlbCwgcHJvcEtleSk7XG4gIGlmICh0eXBlRGVmLm5hbWUgPT09IFwiRnVuY3Rpb25cIikge1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbiAgY29uc3QgbWV0YSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEoXG4gICAgZ2V0SW5qZWN0S2V5KEluamVjdGFibGVzS2V5cy5JTkpFQ1RBQkxFKSxcbiAgICB0eXBlRGVmXG4gICk7XG4gIGlmICghbWV0YSkge1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbiAgcmV0dXJuIG1ldGEuc3ltYm9sO1xufVxuIl19
|
package/lib/utils.d.ts
CHANGED
|
@@ -15,4 +15,4 @@ export declare const TypeKey = "design:type";
|
|
|
15
15
|
* @function getTypeFromDecorator
|
|
16
16
|
* @memberOf module:injectable-decorators
|
|
17
17
|
*/
|
|
18
|
-
export declare function getTypeFromDecorator(model: any, propKey: string | symbol):
|
|
18
|
+
export declare function getTypeFromDecorator(model: any, propKey: string | symbol): symbol | undefined;
|