composed-di 0.0.1
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/README.md +102 -0
- package/dist/index.d.ts +5 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +11 -0
- package/dist/index.js.map +1 -0
- package/dist/main.js +284 -0
- package/dist/serviceFactory.d.ts +25 -0
- package/dist/serviceFactory.d.ts.map +1 -0
- package/dist/serviceFactory.js +44 -0
- package/dist/serviceFactory.js.map +1 -0
- package/dist/serviceKey.d.ts +6 -0
- package/dist/serviceKey.d.ts.map +1 -0
- package/dist/serviceKey.js +12 -0
- package/dist/serviceKey.js.map +1 -0
- package/dist/serviceModule.d.ts +10 -0
- package/dist/serviceModule.d.ts.map +1 -0
- package/dist/serviceModule.js +70 -0
- package/dist/serviceModule.js.map +1 -0
- package/dist/serviceProvider.d.ts +5 -0
- package/dist/serviceProvider.d.ts.map +1 -0
- package/dist/serviceProvider.js +3 -0
- package/dist/serviceProvider.js.map +1 -0
- package/package.json +39 -0
- package/src/index.ts +4 -0
- package/src/serviceFactory.ts +69 -0
- package/src/serviceKey.ts +8 -0
- package/src/serviceModule.ts +91 -0
- package/src/serviceProvider.ts +5 -0
package/README.md
ADDED
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
# composed-di
|
|
2
|
+
|
|
3
|
+
A tiny, type-friendly dependency injection helper for composing services via keys and factories.
|
|
4
|
+
|
|
5
|
+
It provides:
|
|
6
|
+
- ServiceKey<T>: a typed token used to identify a service
|
|
7
|
+
- ServiceFactory<T>: a contract to create a service (with helpers singletonFactory and oneShotFactory)
|
|
8
|
+
- ServiceModule: a resolver that wires factories, validates dependencies, and injects services
|
|
9
|
+
|
|
10
|
+
ServiceModule will:
|
|
11
|
+
- detect recursive dependencies (a factory that depends on its own key)
|
|
12
|
+
- detect missing dependencies (a factory that depends on keys that have no factory)
|
|
13
|
+
|
|
14
|
+
## Install
|
|
15
|
+
|
|
16
|
+
This repository is set up as a library. Build artifacts are generated under `dist/` by running:
|
|
17
|
+
|
|
18
|
+
```
|
|
19
|
+
npm run build
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
## Usage
|
|
23
|
+
|
|
24
|
+
Below is an example based on the previous sample `main.ts` usage pattern.
|
|
25
|
+
|
|
26
|
+
```ts
|
|
27
|
+
import {
|
|
28
|
+
ServiceKey,
|
|
29
|
+
ServiceModule,
|
|
30
|
+
singletonFactory,
|
|
31
|
+
oneShotFactory,
|
|
32
|
+
} from 'composed-di'; // if used locally, import from the relative path to `src/index`
|
|
33
|
+
|
|
34
|
+
// 1) Define service types
|
|
35
|
+
interface Config {
|
|
36
|
+
baseUrl: string;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
interface Logger {
|
|
40
|
+
info: (msg: string) => void;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
class App {
|
|
44
|
+
constructor(private config: Config, private logger: Logger) {}
|
|
45
|
+
start() {
|
|
46
|
+
this.logger.info(`Starting with baseUrl=${this.config.baseUrl}`);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// 2) Create keys
|
|
51
|
+
const ConfigKey = new ServiceKey<Config>('Config');
|
|
52
|
+
const LoggerKey = new ServiceKey<Logger>('Logger');
|
|
53
|
+
const AppKey = new ServiceKey<App>('App');
|
|
54
|
+
|
|
55
|
+
// 3) Create factories (singleton or one-shot)
|
|
56
|
+
const configFactory = singletonFactory({
|
|
57
|
+
provides: ConfigKey,
|
|
58
|
+
dependsOn: [] as const,
|
|
59
|
+
async initialize() {
|
|
60
|
+
return { baseUrl: 'https://api.example.com' } satisfies Config;
|
|
61
|
+
},
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
const loggerFactory = singletonFactory({
|
|
65
|
+
provides: LoggerKey,
|
|
66
|
+
dependsOn: [] as const,
|
|
67
|
+
async initialize() {
|
|
68
|
+
return console as unknown as Logger;
|
|
69
|
+
},
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
const appFactory = oneShotFactory({
|
|
73
|
+
provides: AppKey,
|
|
74
|
+
dependsOn: [ConfigKey, LoggerKey] as const,
|
|
75
|
+
async initialize(config, logger) {
|
|
76
|
+
return new App(config, logger);
|
|
77
|
+
},
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
// 4) Compose a module
|
|
81
|
+
const module = ServiceModule.from([configFactory, loggerFactory, appFactory]);
|
|
82
|
+
|
|
83
|
+
// 5) Inject and use
|
|
84
|
+
(async () => {
|
|
85
|
+
const app = await module.inject(AppKey);
|
|
86
|
+
app.start();
|
|
87
|
+
})();
|
|
88
|
+
```
|
|
89
|
+
|
|
90
|
+
Notes:
|
|
91
|
+
- Use `as const` on your `dependsOn` list to preserve tuple types and keep constructor parameters strongly typed.
|
|
92
|
+
- `ServiceModule.inject` resolves dependencies recursively, so factories can depend on other services.
|
|
93
|
+
- If a dependency is missing or recursive, `ServiceModule` throws with a helpful error message.
|
|
94
|
+
|
|
95
|
+
## API
|
|
96
|
+
|
|
97
|
+
- `class ServiceKey<T>(name: string)`
|
|
98
|
+
- `interface ServiceFactory<T, D extends ServiceKey<unknown>[]>` with `provides`, `dependsOn`, `initialize`, `dispose`
|
|
99
|
+
- `singletonFactory({ provides, dependsOn?, initialize, dispose? })`
|
|
100
|
+
- `oneShotFactory({ provides, dependsOn, initialize, dispose? })`
|
|
101
|
+
- `class ServiceModule` with `static from(factories)`, `inject(key)`
|
|
102
|
+
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export { ServiceKey } from './serviceKey';
|
|
2
|
+
export { ServiceModule } from './serviceModule';
|
|
3
|
+
export { type ServiceFactory, singletonFactory, oneShotFactory } from './serviceFactory';
|
|
4
|
+
export { type ServiceProvider } from './serviceProvider';
|
|
5
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,iBAAiB,CAAC;AAChD,OAAO,EAAE,KAAK,cAAc,EAAE,gBAAgB,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AACzF,OAAO,EAAE,KAAK,eAAe,EAAE,MAAM,mBAAmB,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.oneShotFactory = exports.singletonFactory = exports.ServiceModule = exports.ServiceKey = void 0;
|
|
4
|
+
var serviceKey_1 = require("./serviceKey");
|
|
5
|
+
Object.defineProperty(exports, "ServiceKey", { enumerable: true, get: function () { return serviceKey_1.ServiceKey; } });
|
|
6
|
+
var serviceModule_1 = require("./serviceModule");
|
|
7
|
+
Object.defineProperty(exports, "ServiceModule", { enumerable: true, get: function () { return serviceModule_1.ServiceModule; } });
|
|
8
|
+
var serviceFactory_1 = require("./serviceFactory");
|
|
9
|
+
Object.defineProperty(exports, "singletonFactory", { enumerable: true, get: function () { return serviceFactory_1.singletonFactory; } });
|
|
10
|
+
Object.defineProperty(exports, "oneShotFactory", { enumerable: true, get: function () { return serviceFactory_1.oneShotFactory; } });
|
|
11
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,2CAA0C;AAAjC,wGAAA,UAAU,OAAA;AACnB,iDAAgD;AAAvC,8GAAA,aAAa,OAAA;AACtB,mDAAyF;AAA3D,kHAAA,gBAAgB,OAAA;AAAE,gHAAA,cAAc,OAAA"}
|
package/dist/main.js
ADDED
|
@@ -0,0 +1,284 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __copyProps = (to, from, except, desc) => {
|
|
9
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
10
|
+
for (let key of __getOwnPropNames(from))
|
|
11
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
12
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
13
|
+
}
|
|
14
|
+
return to;
|
|
15
|
+
};
|
|
16
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
17
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
18
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
19
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
20
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
21
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
22
|
+
mod
|
|
23
|
+
));
|
|
24
|
+
|
|
25
|
+
// src/lib/serviceModule.ts
|
|
26
|
+
var ServiceModule = class _ServiceModule {
|
|
27
|
+
constructor(factories) {
|
|
28
|
+
this.factories = [];
|
|
29
|
+
this.factories = Array.from(factories);
|
|
30
|
+
this.factories.forEach((factory) => {
|
|
31
|
+
checkRecursiveDependencies(factory);
|
|
32
|
+
checkMissingDependencies(factory, this.factories);
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
async inject(key) {
|
|
36
|
+
const factory = this.factories.find((factory2) => {
|
|
37
|
+
return isSuitable(key, factory2);
|
|
38
|
+
});
|
|
39
|
+
if (!factory) {
|
|
40
|
+
throw new Error(`Could not find a suitable factory for ${key.name}`);
|
|
41
|
+
}
|
|
42
|
+
const dependencies = await Promise.all(
|
|
43
|
+
factory.dependsOn.map((dependencyKey) => {
|
|
44
|
+
return this.inject(dependencyKey);
|
|
45
|
+
})
|
|
46
|
+
);
|
|
47
|
+
return factory.initialize(...dependencies);
|
|
48
|
+
}
|
|
49
|
+
static from(factories) {
|
|
50
|
+
return new _ServiceModule(new Set(factories));
|
|
51
|
+
}
|
|
52
|
+
};
|
|
53
|
+
function checkRecursiveDependencies(factory) {
|
|
54
|
+
const recursive = factory.dependsOn.some((dependencyKey) => {
|
|
55
|
+
return dependencyKey === factory.provides;
|
|
56
|
+
});
|
|
57
|
+
if (recursive) {
|
|
58
|
+
throw new Error(
|
|
59
|
+
"Recursive dependency detected on: " + factory.provides.name
|
|
60
|
+
);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
function checkMissingDependencies(factory, factories) {
|
|
64
|
+
const missingDependencies = factory.dependsOn.filter(
|
|
65
|
+
(dependencyKey) => {
|
|
66
|
+
return !isRegistered(dependencyKey, factories);
|
|
67
|
+
}
|
|
68
|
+
);
|
|
69
|
+
if (missingDependencies.length === 0) {
|
|
70
|
+
return;
|
|
71
|
+
}
|
|
72
|
+
const dependencyList = missingDependencies.map((dependencyKey) => ` -> ${dependencyKey.name}`).join("\n");
|
|
73
|
+
throw new Error(
|
|
74
|
+
`${factory.provides.name} will fail because it depends on:
|
|
75
|
+
${dependencyList}`
|
|
76
|
+
);
|
|
77
|
+
}
|
|
78
|
+
function isRegistered(key, factories) {
|
|
79
|
+
return factories.some((factory) => factory.provides === key);
|
|
80
|
+
}
|
|
81
|
+
function isSuitable(key, factory) {
|
|
82
|
+
return factory?.provides === key;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// src/lib/serviceKey.ts
|
|
86
|
+
var ServiceKey = class {
|
|
87
|
+
constructor(name) {
|
|
88
|
+
this.name = name;
|
|
89
|
+
this.symbol = Symbol(name);
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
// src/lib/serviceFactory.ts
|
|
94
|
+
function singletonFactory({
|
|
95
|
+
provides,
|
|
96
|
+
dependsOn = [],
|
|
97
|
+
initialize,
|
|
98
|
+
dispose = () => {
|
|
99
|
+
}
|
|
100
|
+
}) {
|
|
101
|
+
let instance;
|
|
102
|
+
return {
|
|
103
|
+
provides,
|
|
104
|
+
dependsOn,
|
|
105
|
+
async initialize(...dependencies) {
|
|
106
|
+
if (instance) {
|
|
107
|
+
return instance;
|
|
108
|
+
}
|
|
109
|
+
instance = await initialize(...dependencies);
|
|
110
|
+
return instance;
|
|
111
|
+
},
|
|
112
|
+
dispose(serviceInstance) {
|
|
113
|
+
if (instance === serviceInstance) {
|
|
114
|
+
dispose(serviceInstance);
|
|
115
|
+
instance = void 0;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// src/core/sms/di/smsModule.ts
|
|
122
|
+
var import_node_process = __toESM(require("node:process"));
|
|
123
|
+
|
|
124
|
+
// src/core/sms/noOpSmsService.ts
|
|
125
|
+
var NoOpSmsService = class {
|
|
126
|
+
send(recipient, text) {
|
|
127
|
+
console.warn(`NoOpSmsService: Sending sms to ${recipient}: ${text}`);
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
// src/core/sms/tigoSmsService.ts
|
|
132
|
+
var TigoSmsService = class {
|
|
133
|
+
send(_recipient, _text) {
|
|
134
|
+
throw new Error("Not yet implemented");
|
|
135
|
+
}
|
|
136
|
+
};
|
|
137
|
+
|
|
138
|
+
// src/core/logger/emailAlertLogger.ts
|
|
139
|
+
var EmailAlertLogger = class {
|
|
140
|
+
constructor(bufferSize = 50, recipients = []) {
|
|
141
|
+
this.buffer = [];
|
|
142
|
+
this.emailQueue = [];
|
|
143
|
+
this.bufferSize = bufferSize;
|
|
144
|
+
this.recipients = recipients;
|
|
145
|
+
}
|
|
146
|
+
addToBuffer(event) {
|
|
147
|
+
if (this.buffer.length >= this.bufferSize) {
|
|
148
|
+
this.buffer.shift();
|
|
149
|
+
}
|
|
150
|
+
this.buffer.push({ ...event, date: /* @__PURE__ */ new Date() });
|
|
151
|
+
}
|
|
152
|
+
info(message, payload) {
|
|
153
|
+
this.addToBuffer({ level: "INFO", message, payload });
|
|
154
|
+
}
|
|
155
|
+
error(message, error, payload) {
|
|
156
|
+
this.addToBuffer({ level: "ERROR", message, payload });
|
|
157
|
+
this.emailQueue.push(createTableString(this.buffer, error));
|
|
158
|
+
this.buffer.length = 0;
|
|
159
|
+
}
|
|
160
|
+
debug(message, payload) {
|
|
161
|
+
this.addToBuffer({ level: "DEBUG", message, payload });
|
|
162
|
+
}
|
|
163
|
+
async flush() {
|
|
164
|
+
for (const email of this.emailQueue) {
|
|
165
|
+
console.error(email);
|
|
166
|
+
}
|
|
167
|
+
this.buffer.length = 0;
|
|
168
|
+
this.emailQueue.length = 0;
|
|
169
|
+
}
|
|
170
|
+
};
|
|
171
|
+
function createTableString(events, error) {
|
|
172
|
+
if (events.length === 0) {
|
|
173
|
+
return "No events to display";
|
|
174
|
+
}
|
|
175
|
+
const dateWidth = Math.max(
|
|
176
|
+
4,
|
|
177
|
+
...events.map((e) => e.date.toISOString().length)
|
|
178
|
+
);
|
|
179
|
+
const levelWidth = Math.max(
|
|
180
|
+
5,
|
|
181
|
+
...events.map((e) => e.level.length)
|
|
182
|
+
);
|
|
183
|
+
const messageWidth = Math.max(7, ...events.map((e) => e.message.length));
|
|
184
|
+
const payloadWidth = Math.max(
|
|
185
|
+
7,
|
|
186
|
+
...events.map((e) => e.payload ? JSON.stringify(e.payload).length : 0)
|
|
187
|
+
);
|
|
188
|
+
const header = [
|
|
189
|
+
"DATE".padEnd(dateWidth),
|
|
190
|
+
"LEVEL".padEnd(levelWidth),
|
|
191
|
+
"MESSAGE".padEnd(messageWidth),
|
|
192
|
+
"PAYLOAD".padEnd(payloadWidth)
|
|
193
|
+
].join(" | ");
|
|
194
|
+
const separator = [
|
|
195
|
+
"-".repeat(dateWidth),
|
|
196
|
+
"-".repeat(levelWidth),
|
|
197
|
+
"-".repeat(messageWidth),
|
|
198
|
+
"-".repeat(payloadWidth)
|
|
199
|
+
].join("-|-");
|
|
200
|
+
const rows = events.map((event) => {
|
|
201
|
+
const date = event.date.toISOString().padEnd(dateWidth);
|
|
202
|
+
const level = event.level.padEnd(levelWidth);
|
|
203
|
+
const message = event.message.padEnd(messageWidth);
|
|
204
|
+
const payload = (event.payload ? JSON.stringify(event.payload) : "").padEnd(
|
|
205
|
+
payloadWidth
|
|
206
|
+
);
|
|
207
|
+
return [date, level, message, payload].join(" | ");
|
|
208
|
+
});
|
|
209
|
+
const errorSection = error ? "\n\nError Details:\n" + "-".repeat(20) + "\n" + (error instanceof Error ? `${error.name}: ${error.message}
|
|
210
|
+
${error.stack || ""}` : String(error)) : "";
|
|
211
|
+
return [header, separator, ...rows].join("\n") + errorSection;
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
// src/core/logger/di/loggerModule.ts
|
|
215
|
+
var LOGGER = new ServiceKey("Logger");
|
|
216
|
+
var LoggerFactory = singletonFactory({
|
|
217
|
+
provides: LOGGER,
|
|
218
|
+
dependsOn: [],
|
|
219
|
+
initialize: () => {
|
|
220
|
+
return Promise.resolve(new EmailAlertLogger(100, ["juanhr454@gmail.com"]));
|
|
221
|
+
}
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
// src/core/sms/di/smsModule.ts
|
|
225
|
+
var SMS_SERVICE = new ServiceKey("SmsService");
|
|
226
|
+
var SmsServiceFactory = singletonFactory({
|
|
227
|
+
provides: SMS_SERVICE,
|
|
228
|
+
dependsOn: [LOGGER],
|
|
229
|
+
initialize(logger) {
|
|
230
|
+
logger.info("Initializing SmsService");
|
|
231
|
+
if (import_node_process.default.env.environment === "dev") {
|
|
232
|
+
return Promise.resolve(new NoOpSmsService());
|
|
233
|
+
} else {
|
|
234
|
+
return Promise.resolve(new TigoSmsService());
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
});
|
|
238
|
+
var SmsServiceTestFactory = singletonFactory({
|
|
239
|
+
provides: SMS_SERVICE,
|
|
240
|
+
dependsOn: [],
|
|
241
|
+
initialize() {
|
|
242
|
+
return Promise.resolve(new NoOpSmsService());
|
|
243
|
+
}
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
// src/main.ts
|
|
247
|
+
var authModule = ServiceModule.from([SmsServiceFactory, LoggerFactory]);
|
|
248
|
+
function module2(module3, fn) {
|
|
249
|
+
return (event, context) => {
|
|
250
|
+
fn(event, { services: module3, ...context });
|
|
251
|
+
};
|
|
252
|
+
}
|
|
253
|
+
var handler = module2(authModule, login);
|
|
254
|
+
(async function() {
|
|
255
|
+
await deeplyNestedFunction({ services: authModule });
|
|
256
|
+
const logger = await authModule.inject(LOGGER);
|
|
257
|
+
if (logger instanceof EmailAlertLogger) {
|
|
258
|
+
await logger.flush();
|
|
259
|
+
}
|
|
260
|
+
console.log("Done");
|
|
261
|
+
})();
|
|
262
|
+
async function deeplyNestedFunction(context) {
|
|
263
|
+
await login({}, context);
|
|
264
|
+
}
|
|
265
|
+
async function login(event, context) {
|
|
266
|
+
const state = authorize(event);
|
|
267
|
+
const logger = await context.services.inject(LOGGER);
|
|
268
|
+
if (state === "DEVICE_CHANGED") {
|
|
269
|
+
logger.info("Device changed", { "user": "asad" });
|
|
270
|
+
const sms = await context.services.inject(SMS_SERVICE);
|
|
271
|
+
logger.debug("Sending sms");
|
|
272
|
+
try {
|
|
273
|
+
sms.send(971233149, "Hola Juan, tu pin es 22222");
|
|
274
|
+
} catch (e) {
|
|
275
|
+
logger.error("Error sending sms", e);
|
|
276
|
+
}
|
|
277
|
+
const sms1 = await context.services.inject(SMS_SERVICE);
|
|
278
|
+
}
|
|
279
|
+
return;
|
|
280
|
+
}
|
|
281
|
+
function authorize(_request) {
|
|
282
|
+
return "DEVICE_CHANGED";
|
|
283
|
+
}
|
|
284
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/lib/serviceModule.ts", "../src/lib/serviceKey.ts", "../src/lib/serviceFactory.ts", "../src/core/sms/di/smsModule.ts", "../src/core/sms/noOpSmsService.ts", "../src/core/sms/tigoSmsService.ts", "../src/core/logger/emailAlertLogger.ts", "../src/core/logger/di/loggerModule.ts", "../src/main.ts"],
  "sourcesContent": ["import { ServiceKey } from './serviceKey';\r\nimport { ServiceFactory } from './serviceFactory';\r\nimport { ServiceProvider } from './serviceProvider';\r\n\r\nexport class ServiceModule implements ServiceProvider {\r\n  private readonly factories: ServiceFactory<any, any>[] = [];\r\n\r\n  constructor(factories: Set<ServiceFactory<unknown, ServiceKey<unknown>[]>>) {\r\n    this.factories = Array.from(factories);\r\n    this.factories.forEach((factory) => {\r\n      checkRecursiveDependencies(factory);\r\n      checkMissingDependencies(factory, this.factories);\r\n    });\r\n  }\r\n\r\n  public async inject<T>(key: ServiceKey<T>): Promise<T> {\r\n    const factory = this.factories.find((factory) => {\r\n      return isSuitable(key, factory);\r\n    });\r\n\r\n    // Check if a factory to supply the requested key was not found\r\n    if (!factory) {\r\n      throw new Error(`Could not find a suitable factory for ${key.name}`);\r\n    }\r\n\r\n    // Resolve all dependencies first\r\n    const dependencies = await Promise.all(\r\n      factory.dependsOn.map((dependencyKey: ServiceKey<unknown>) => {\r\n        return this.inject(dependencyKey);\r\n      }),\r\n    );\r\n\r\n    // Call the factory to retrieve the dependency\r\n    return factory.initialize(...dependencies);\r\n  }\r\n\r\n  static from(\r\n    factories: ServiceFactory<unknown, ServiceKey<unknown>[]>[],\r\n  ): ServiceModule {\r\n    return new ServiceModule(new Set(factories));\r\n  }\r\n}\r\n\r\nfunction checkRecursiveDependencies(\r\n  factory: ServiceFactory<unknown, ServiceKey<unknown>[]>,\r\n) {\r\n  const recursive = factory.dependsOn.some((dependencyKey) => {\r\n    return dependencyKey === factory.provides;\r\n  });\r\n\r\n  if (recursive) {\r\n    throw new Error(\r\n      'Recursive dependency detected on: ' + factory.provides.name,\r\n    );\r\n  }\r\n}\r\n\r\nfunction checkMissingDependencies(\r\n  factory: ServiceFactory<unknown, ServiceKey<unknown>[]>,\r\n  factories: ServiceFactory<unknown>[],\r\n) {\r\n  const missingDependencies = factory.dependsOn.filter(\r\n    (dependencyKey: ServiceKey<any>) => {\r\n      return !isRegistered(dependencyKey, factories);\r\n    },\r\n  );\r\n  if (missingDependencies.length === 0) {\r\n    return;\r\n  }\r\n\r\n  const dependencyList = missingDependencies\r\n    .map((dependencyKey) => ` -> ${dependencyKey.name}`)\r\n    .join('\\n');\r\n  throw new Error(\r\n    `${factory.provides.name} will fail because it depends on:\\n ${dependencyList}`,\r\n  );\r\n}\r\n\r\nfunction isRegistered(\r\n  key: ServiceKey<unknown>,\r\n  factories: ServiceFactory<unknown>[],\r\n) {\r\n  return factories.some((factory) => factory.provides === key);\r\n}\r\n\r\nfunction isSuitable<T, D extends ServiceKey<unknown>[]>(\r\n  key: ServiceKey<T>,\r\n  factory: ServiceFactory<unknown, D>,\r\n): factory is ServiceFactory<T, D> {\r\n  return factory?.provides === key;\r\n}\r\n", "// @ts-ignore\r\nexport class ServiceKey<T> {\r\n  private readonly symbol: symbol;\r\n\r\n  constructor(public readonly name: string) {\r\n    this.symbol = Symbol(name);\r\n  }\r\n}\r\n", "import { ServiceKey } from './serviceKey';\r\n\r\n// Helper types to extract the type from ServiceKey\r\ntype ServiceType<T> = T extends ServiceKey<infer U> ? U : never;\r\n\r\n// Helper types to convert an array of ServiceKey to tuple of their types\r\ntype DependencyTypes<T extends ServiceKey<unknown>[]> = {\r\n  [K in keyof T]: ServiceType<T[K]>;\r\n};\r\n\r\nexport interface ServiceFactory<T, D extends ServiceKey<unknown>[] = []> {\r\n  provides: ServiceKey<T>;\r\n  dependsOn: D;\r\n\r\n  initialize(...dependencies: DependencyTypes<D>): Promise<T>;\r\n\r\n  dispose(instance: T): void;\r\n}\r\n\r\nexport function singletonFactory<T, D extends ServiceKey<unknown>[] = []>({\r\n  provides,\r\n  dependsOn = [] as unknown as D,\r\n  initialize,\r\n  dispose = () => {},\r\n}: {\r\n  provides: ServiceKey<T>;\r\n  dependsOn?: D;\r\n  initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;\r\n  dispose?: (instance: T) => void;\r\n}): ServiceFactory<T, D> {\r\n  let instance: T | undefined;\r\n\r\n  return {\r\n    provides,\r\n    dependsOn,\r\n    async initialize(...dependencies: DependencyTypes<D>): Promise<T> {\r\n      if (instance) {\r\n        return instance;\r\n      }\r\n      instance = await initialize(...dependencies);\r\n      return instance;\r\n    },\r\n    dispose(serviceInstance: T): void {\r\n      if (instance === serviceInstance) {\r\n        dispose(serviceInstance);\r\n        instance = undefined;\r\n      }\r\n    },\r\n  };\r\n}\r\n\r\nexport function oneShotFactory<T, D extends ServiceKey<unknown>[] = []>({\r\n  provides,\r\n  dependsOn,\r\n  initialize,\r\n  dispose = () => {},\r\n}: {\r\n  provides: ServiceKey<T>;\r\n  dependsOn: D;\r\n  initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;\r\n  dispose?: (instance: T) => void;\r\n}): ServiceFactory<T, D> {\r\n  return {\r\n    provides,\r\n    dependsOn,\r\n    initialize,\r\n    dispose,\r\n  };\r\n}\r\n", "import { ServiceKey } from '../../../lib/serviceKey';\nimport { SmsService } from '../smsService';\nimport { singletonFactory } from '../../../lib/serviceFactory';\nimport process from 'node:process';\nimport { NoOpSmsService } from '../noOpSmsService';\nimport { TigoSmsService } from '../tigoSmsService';\nimport { LOGGER } from '../../logger/di/loggerModule';\nimport { Logger } from '../../logger/logger';\n\nexport const SMS_SERVICE = new ServiceKey<SmsService>('SmsService');\n\nexport const SmsServiceFactory = singletonFactory({\n  provides: SMS_SERVICE,\n  dependsOn: [LOGGER],\n  initialize(logger: Logger): Promise<SmsService> {\n    logger.info('Initializing SmsService');\n    // Since the create method is async, we could instantiate a different\n    // implementation on demand by retrieving some configuration from a db\n    // or a config file.\n    if (process.env.environment === 'dev') {\n      return Promise.resolve(new NoOpSmsService());\n    } else {\n      return Promise.resolve(new TigoSmsService());\n    }\n  },\n});\n\nexport const SmsServiceTestFactory = singletonFactory({\n  provides: SMS_SERVICE,\n  dependsOn: [],\n  initialize(): Promise<SmsService> {\n    return Promise.resolve(new NoOpSmsService());\n  },\n})", "import { SmsService } from './smsService';\r\n\r\nexport class NoOpSmsService implements SmsService {\r\n  send(recipient: number, text: string) {\r\n    console.warn(`NoOpSmsService: Sending sms to ${recipient}: ${text}`);\r\n  }\r\n}\r\n", "import { SmsService } from './smsService';\r\n\r\nexport class TigoSmsService implements SmsService {\r\n  send(_recipient: number, _text: string) {\r\n    throw new Error('Not yet implemented');\r\n  }\r\n}\r\n", "import { Logger, LogEvent, LogEventPayload } from './logger';\r\n\r\nexport class EmailAlertLogger implements Logger {\r\n  private readonly buffer: LogEvent[] = [];\r\n  private readonly bufferSize: number;\r\n  private readonly recipients: string[];\r\n  private readonly emailQueue: string[] = [];\r\n\r\n  constructor(bufferSize: number = 50, recipients: string[] = []) {\r\n    this.bufferSize = bufferSize;\r\n    this.recipients = recipients;\r\n  }\r\n\r\n  private addToBuffer(event: Omit<LogEvent, 'date'>): void {\r\n    // Keep the buffer to the specified size\r\n    if (this.buffer.length >= this.bufferSize) {\r\n      this.buffer.shift();\r\n    }\r\n    this.buffer.push({ ...event, date: new Date() });\r\n  }\r\n\r\n  public info(message: string, payload?: LogEventPayload): void {\r\n    this.addToBuffer({ level: 'INFO', message, payload });\r\n  }\r\n\r\n  public error(\r\n    message: string,\r\n    error?: unknown,\r\n    payload?: LogEventPayload,\r\n  ): void {\r\n    this.addToBuffer({ level: 'ERROR', message, payload });\r\n    this.emailQueue.push(createTableString(this.buffer, error));\r\n    this.buffer.length = 0;\r\n  }\r\n\r\n  public debug(message: string, payload?: LogEventPayload): void {\r\n    this.addToBuffer({ level: 'DEBUG', message, payload });\r\n  }\r\n\r\n  public async flush(): Promise<void> {\r\n    for (const email of this.emailQueue) {\r\n      console.error(email);\r\n    }\r\n    // Clear all buffers\r\n    this.buffer.length = 0;\r\n    this.emailQueue.length = 0;\r\n  }\r\n}\r\n\r\nfunction createTableString(events: LogEvent[], error?: unknown): string {\r\n  if (events.length === 0) {\r\n    return 'No events to display';\r\n  }\r\n\r\n  // Calculate column widths\r\n  const dateWidth = Math.max(\r\n    4,\r\n    ...events.map((e) => e.date.toISOString().length),\r\n  );\r\n  const levelWidth = Math.max(\r\n    5, // \"Level\" header length\r\n    ...events.map((e) => e.level.length),\r\n  );\r\n  const messageWidth = Math.max(7, ...events.map((e) => e.message.length));\r\n  const payloadWidth = Math.max(\r\n    7,\r\n    ...events.map((e) => (e.payload ? JSON.stringify(e.payload).length : 0)),\r\n  );\r\n\r\n  // Create header\r\n  const header = [\r\n    'DATE'.padEnd(dateWidth),\r\n    'LEVEL'.padEnd(levelWidth),\r\n    'MESSAGE'.padEnd(messageWidth),\r\n    'PAYLOAD'.padEnd(payloadWidth),\r\n  ].join(' | ');\r\n\r\n  // Create separator\r\n  const separator = [\r\n    '-'.repeat(dateWidth),\r\n    '-'.repeat(levelWidth),\r\n    '-'.repeat(messageWidth),\r\n    '-'.repeat(payloadWidth),\r\n  ].join('-|-');\r\n\r\n  // Create rows\r\n  const rows = events.map((event) => {\r\n    const date = event.date.toISOString().padEnd(dateWidth);\r\n    const level = event.level.padEnd(levelWidth);\r\n    const message = event.message.padEnd(messageWidth);\r\n    const payload = (event.payload ? JSON.stringify(event.payload) : '').padEnd(\r\n      payloadWidth,\r\n    );\r\n    return [date, level, message, payload].join(' | ');\r\n  });\r\n\r\n  // Create error section if error exists\r\n  const errorSection = error\r\n    ? '\\n\\nError Details:\\n' +\r\n      '-'.repeat(20) +\r\n      '\\n' +\r\n      (error instanceof Error\r\n        ? `${error.name}: ${error.message}\\n${error.stack || ''}`\r\n        : String(error))\r\n    : '';\r\n\r\n  // Combine all parts\r\n  return [header, separator, ...rows].join('\\n') + errorSection;\r\n}\r\n", "import { singletonFactory } from '../../../lib/serviceFactory';\r\nimport { ServiceKey } from '../../../lib/serviceKey';\r\nimport { EmailAlertLogger } from '../emailAlertLogger';\r\nimport { Logger } from '../logger';\r\nimport { SMS_SERVICE } from '../../sms/di/smsModule';\r\nimport { SmsService } from '../../sms/smsService';\r\n\r\nexport const LOGGER = new ServiceKey<Logger>('Logger');\r\n\r\nexport const LoggerFactory = singletonFactory({\r\n  provides: LOGGER,\r\n  dependsOn: [],\r\n  initialize: () => {\r\n    return Promise.resolve(new EmailAlertLogger(100, ['juanhr454@gmail.com']));\r\n  },\r\n});\r\n", "import { ServiceModule } from './lib/serviceModule';\r\nimport { ServiceProvider } from './lib/serviceProvider';\r\nimport { SMS_SERVICE, SmsServiceFactory } from './core/sms/di/smsModule';\r\nimport { LOGGER, LoggerFactory } from './core/logger/di/loggerModule';\r\nimport { EmailAlertLogger } from './core/logger/emailAlertLogger';\r\n\r\nconst authModule = ServiceModule.from([SmsServiceFactory, LoggerFactory]);\r\n\r\nfunction module(\r\n  module: ServiceModule,\r\n  fn: (event: object, context: Context) => void,\r\n) {\r\n  return (event: object, context: object) => {\r\n    fn(event, { services: module, ...context });\r\n  };\r\n}\r\n\r\nconst handler = module(authModule, login);\r\n\r\n(async function () {\r\n  await deeplyNestedFunction({ services: authModule });\r\n  const logger = await authModule.inject(LOGGER);\r\n  if (logger instanceof EmailAlertLogger) {\r\n    await logger.flush();\r\n  }\r\n  console.log('Done');\r\n})();\r\n\r\nasync function deeplyNestedFunction(context: Context) {\r\n  await login({}, context);\r\n}\r\n\r\ninterface Context {\r\n  services: ServiceProvider;\r\n}\r\n\r\nasync function login(event: any, context: Context) {\r\n  const state = authorize(event);\r\n  const logger = await context.services.inject(LOGGER);\r\n  if (state === 'DEVICE_CHANGED') {\r\n    logger.info('Device changed', {'user': 'asad'});\r\n    // With the service key we get type safety/inference and navigate to definition for free,\r\n    // there is no magic no annotations, we can see all the\r\n    // factories that provide this dependency and see all places where this is retrieved.\r\n    const sms = await context.services.inject(SMS_SERVICE);\r\n    logger.debug('Sending sms');\r\n    try {\r\n      sms.send(971233149, 'Hola Juan, tu pin es 22222');\r\n    } catch (e) {\r\n      logger.error('Error sending sms', e);\r\n    }\r\n\r\n    // Retrieving the service a second time would have virtually zero cost if the\r\n    // underlying factory provides a singleton.\r\n    const sms1 = await context.services.inject(SMS_SERVICE);\r\n  }\r\n\r\n  return;\r\n}\r\n\r\nfunction authorize(_request: any): 'OK' | 'DEVICE_CHANGED' {\r\n  return 'DEVICE_CHANGED';\r\n}\r\n"],
  "mappings": "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",
  "names": ["factory", "process", "module"]
}

|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { ServiceKey } from './serviceKey';
|
|
2
|
+
type ServiceType<T> = T extends ServiceKey<infer U> ? U : never;
|
|
3
|
+
type DependencyTypes<T extends ServiceKey<unknown>[]> = {
|
|
4
|
+
[K in keyof T]: ServiceType<T[K]>;
|
|
5
|
+
};
|
|
6
|
+
export interface ServiceFactory<T, D extends ServiceKey<unknown>[] = []> {
|
|
7
|
+
provides: ServiceKey<T>;
|
|
8
|
+
dependsOn: D;
|
|
9
|
+
initialize(...dependencies: DependencyTypes<D>): Promise<T>;
|
|
10
|
+
dispose(instance: T): void;
|
|
11
|
+
}
|
|
12
|
+
export declare function singletonFactory<T, D extends ServiceKey<unknown>[] = []>({ provides, dependsOn, initialize, dispose, }: {
|
|
13
|
+
provides: ServiceKey<T>;
|
|
14
|
+
dependsOn?: D;
|
|
15
|
+
initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;
|
|
16
|
+
dispose?: (instance: T) => void;
|
|
17
|
+
}): ServiceFactory<T, D>;
|
|
18
|
+
export declare function oneShotFactory<T, D extends ServiceKey<unknown>[] = []>({ provides, dependsOn, initialize, dispose, }: {
|
|
19
|
+
provides: ServiceKey<T>;
|
|
20
|
+
dependsOn: D;
|
|
21
|
+
initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;
|
|
22
|
+
dispose?: (instance: T) => void;
|
|
23
|
+
}): ServiceFactory<T, D>;
|
|
24
|
+
export {};
|
|
25
|
+
//# sourceMappingURL=serviceFactory.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceFactory.d.ts","sourceRoot":"","sources":["../src/serviceFactory.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAG1C,KAAK,WAAW,CAAC,CAAC,IAAI,CAAC,SAAS,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;AAGhE,KAAK,eAAe,CAAC,CAAC,SAAS,UAAU,CAAC,OAAO,CAAC,EAAE,IAAI;KACrD,CAAC,IAAI,MAAM,CAAC,GAAG,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;CAClC,CAAC;AAEF,MAAM,WAAW,cAAc,CAAC,CAAC,EAAE,CAAC,SAAS,UAAU,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE;IACrE,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC;IACxB,SAAS,EAAE,CAAC,CAAC;IAEb,UAAU,CAAC,GAAG,YAAY,EAAE,eAAe,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;IAE5D,OAAO,CAAC,QAAQ,EAAE,CAAC,GAAG,IAAI,CAAC;CAC5B;AAED,wBAAgB,gBAAgB,CAAC,CAAC,EAAE,CAAC,SAAS,UAAU,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE,EAAE,EACxE,QAAQ,EACR,SAA8B,EAC9B,UAAU,EACV,OAAkB,GACnB,EAAE;IACD,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC;IACxB,SAAS,CAAC,EAAE,CAAC,CAAC;IACd,UAAU,EAAE,CAAC,GAAG,YAAY,EAAE,eAAe,CAAC,CAAC,CAAC,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC;IAChE,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,KAAK,IAAI,CAAC;CACjC,GAAG,cAAc,CAAC,CAAC,EAAE,CAAC,CAAC,CAoBvB;AAED,wBAAgB,cAAc,CAAC,CAAC,EAAE,CAAC,SAAS,UAAU,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE,EAAE,EACtE,QAAQ,EACR,SAAS,EACT,UAAU,EACV,OAAkB,GACnB,EAAE;IACD,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC;IACxB,SAAS,EAAE,CAAC,CAAC;IACb,UAAU,EAAE,CAAC,GAAG,YAAY,EAAE,eAAe,CAAC,CAAC,CAAC,KAAK,OAAO,CAAC,CAAC,CAAC,CAAC;IAChE,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,KAAK,IAAI,CAAC;CACjC,GAAG,cAAc,CAAC,CAAC,EAAE,CAAC,CAAC,CAOvB"}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.singletonFactory = singletonFactory;
|
|
13
|
+
exports.oneShotFactory = oneShotFactory;
|
|
14
|
+
function singletonFactory({ provides, dependsOn = [], initialize, dispose = () => { }, }) {
|
|
15
|
+
let instance;
|
|
16
|
+
return {
|
|
17
|
+
provides,
|
|
18
|
+
dependsOn,
|
|
19
|
+
initialize(...dependencies) {
|
|
20
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
21
|
+
if (instance) {
|
|
22
|
+
return instance;
|
|
23
|
+
}
|
|
24
|
+
instance = yield initialize(...dependencies);
|
|
25
|
+
return instance;
|
|
26
|
+
});
|
|
27
|
+
},
|
|
28
|
+
dispose(serviceInstance) {
|
|
29
|
+
if (instance === serviceInstance) {
|
|
30
|
+
dispose(serviceInstance);
|
|
31
|
+
instance = undefined;
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
function oneShotFactory({ provides, dependsOn, initialize, dispose = () => { }, }) {
|
|
37
|
+
return {
|
|
38
|
+
provides,
|
|
39
|
+
dependsOn,
|
|
40
|
+
initialize,
|
|
41
|
+
dispose,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=serviceFactory.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceFactory.js","sourceRoot":"","sources":["../src/serviceFactory.ts"],"names":[],"mappings":";;;;;;;;;;;AAmBA,4CA8BC;AAED,wCAiBC;AAjDD,SAAgB,gBAAgB,CAA0C,EACxE,QAAQ,EACR,SAAS,GAAG,EAAkB,EAC9B,UAAU,EACV,OAAO,GAAG,GAAG,EAAE,GAAE,CAAC,GAMnB;IACC,IAAI,QAAuB,CAAC;IAE5B,OAAO;QACL,QAAQ;QACR,SAAS;QACH,UAAU,CAAC,GAAG,YAAgC;;gBAClD,IAAI,QAAQ,EAAE,CAAC;oBACb,OAAO,QAAQ,CAAC;gBAClB,CAAC;gBACD,QAAQ,GAAG,MAAM,UAAU,CAAC,GAAG,YAAY,CAAC,CAAC;gBAC7C,OAAO,QAAQ,CAAC;YAClB,CAAC;SAAA;QACD,OAAO,CAAC,eAAkB;YACxB,IAAI,QAAQ,KAAK,eAAe,EAAE,CAAC;gBACjC,OAAO,CAAC,eAAe,CAAC,CAAC;gBACzB,QAAQ,GAAG,SAAS,CAAC;YACvB,CAAC;QACH,CAAC;KACF,CAAC;AACJ,CAAC;AAED,SAAgB,cAAc,CAA0C,EACtE,QAAQ,EACR,SAAS,EACT,UAAU,EACV,OAAO,GAAG,GAAG,EAAE,GAAE,CAAC,GAMnB;IACC,OAAO;QACL,QAAQ;QACR,SAAS;QACT,UAAU;QACV,OAAO;KACR,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceKey.d.ts","sourceRoot":"","sources":["../src/serviceKey.ts"],"names":[],"mappings":"AACA,qBAAa,UAAU,CAAC,CAAC;aAGK,IAAI,EAAE,MAAM;IAFxC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;gBAEJ,IAAI,EAAE,MAAM;CAGzC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ServiceKey = void 0;
|
|
4
|
+
// @ts-ignore
|
|
5
|
+
class ServiceKey {
|
|
6
|
+
constructor(name) {
|
|
7
|
+
this.name = name;
|
|
8
|
+
this.symbol = Symbol(name);
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
exports.ServiceKey = ServiceKey;
|
|
12
|
+
//# sourceMappingURL=serviceKey.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceKey.js","sourceRoot":"","sources":["../src/serviceKey.ts"],"names":[],"mappings":";;;AAAA,aAAa;AACb,MAAa,UAAU;IAGrB,YAA4B,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;QACtC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;IAC7B,CAAC;CACF;AAND,gCAMC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { ServiceKey } from './serviceKey';
|
|
2
|
+
import { ServiceFactory } from './serviceFactory';
|
|
3
|
+
import { ServiceProvider } from './serviceProvider';
|
|
4
|
+
export declare class ServiceModule implements ServiceProvider {
|
|
5
|
+
private readonly factories;
|
|
6
|
+
constructor(factories: Set<ServiceFactory<unknown, ServiceKey<unknown>[]>>);
|
|
7
|
+
inject<T>(key: ServiceKey<T>): Promise<T>;
|
|
8
|
+
static from(factories: ServiceFactory<unknown, ServiceKey<unknown>[]>[]): ServiceModule;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=serviceModule.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceModule.d.ts","sourceRoot":"","sources":["../src/serviceModule.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEpD,qBAAa,aAAc,YAAW,eAAe;IACnD,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAkC;gBAEhD,SAAS,EAAE,GAAG,CAAC,cAAc,CAAC,OAAO,EAAE,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IAQ7D,MAAM,CAAC,CAAC,EAAE,GAAG,EAAE,UAAU,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAqBtD,MAAM,CAAC,IAAI,CACT,SAAS,EAAE,cAAc,CAAC,OAAO,EAAE,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,GAC1D,aAAa;CAGjB"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.ServiceModule = void 0;
|
|
13
|
+
class ServiceModule {
|
|
14
|
+
constructor(factories) {
|
|
15
|
+
this.factories = [];
|
|
16
|
+
this.factories = Array.from(factories);
|
|
17
|
+
this.factories.forEach((factory) => {
|
|
18
|
+
checkRecursiveDependencies(factory);
|
|
19
|
+
checkMissingDependencies(factory, this.factories);
|
|
20
|
+
});
|
|
21
|
+
}
|
|
22
|
+
inject(key) {
|
|
23
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
24
|
+
const factory = this.factories.find((factory) => {
|
|
25
|
+
return isSuitable(key, factory);
|
|
26
|
+
});
|
|
27
|
+
// Check if a factory to supply the requested key was not found
|
|
28
|
+
if (!factory) {
|
|
29
|
+
throw new Error(`Could not find a suitable factory for ${key.name}`);
|
|
30
|
+
}
|
|
31
|
+
// Resolve all dependencies first
|
|
32
|
+
const dependencies = yield Promise.all(factory.dependsOn.map((dependencyKey) => {
|
|
33
|
+
return this.inject(dependencyKey);
|
|
34
|
+
}));
|
|
35
|
+
// Call the factory to retrieve the dependency
|
|
36
|
+
return factory.initialize(...dependencies);
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
static from(factories) {
|
|
40
|
+
return new ServiceModule(new Set(factories));
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
exports.ServiceModule = ServiceModule;
|
|
44
|
+
function checkRecursiveDependencies(factory) {
|
|
45
|
+
const recursive = factory.dependsOn.some((dependencyKey) => {
|
|
46
|
+
return dependencyKey === factory.provides;
|
|
47
|
+
});
|
|
48
|
+
if (recursive) {
|
|
49
|
+
throw new Error('Recursive dependency detected on: ' + factory.provides.name);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
function checkMissingDependencies(factory, factories) {
|
|
53
|
+
const missingDependencies = factory.dependsOn.filter((dependencyKey) => {
|
|
54
|
+
return !isRegistered(dependencyKey, factories);
|
|
55
|
+
});
|
|
56
|
+
if (missingDependencies.length === 0) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const dependencyList = missingDependencies
|
|
60
|
+
.map((dependencyKey) => ` -> ${dependencyKey.name}`)
|
|
61
|
+
.join('\n');
|
|
62
|
+
throw new Error(`${factory.provides.name} will fail because it depends on:\n ${dependencyList}`);
|
|
63
|
+
}
|
|
64
|
+
function isRegistered(key, factories) {
|
|
65
|
+
return factories.some((factory) => factory.provides === key);
|
|
66
|
+
}
|
|
67
|
+
function isSuitable(key, factory) {
|
|
68
|
+
return (factory === null || factory === void 0 ? void 0 : factory.provides) === key;
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=serviceModule.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceModule.js","sourceRoot":"","sources":["../src/serviceModule.ts"],"names":[],"mappings":";;;;;;;;;;;;AAIA,MAAa,aAAa;IAGxB,YAAY,SAA8D;QAFzD,cAAS,GAA+B,EAAE,CAAC;QAG1D,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YACjC,0BAA0B,CAAC,OAAO,CAAC,CAAC;YACpC,wBAAwB,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;IACL,CAAC;IAEY,MAAM,CAAI,GAAkB;;YACvC,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE;gBAC9C,OAAO,UAAU,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;YAClC,CAAC,CAAC,CAAC;YAEH,+DAA+D;YAC/D,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,MAAM,IAAI,KAAK,CAAC,yCAAyC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;YACvE,CAAC;YAED,iCAAiC;YACjC,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,GAAG,CACpC,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,aAAkC,EAAE,EAAE;gBAC3D,OAAO,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;YACpC,CAAC,CAAC,CACH,CAAC;YAEF,8CAA8C;YAC9C,OAAO,OAAO,CAAC,UAAU,CAAC,GAAG,YAAY,CAAC,CAAC;QAC7C,CAAC;KAAA;IAED,MAAM,CAAC,IAAI,CACT,SAA2D;QAE3D,OAAO,IAAI,aAAa,CAAC,IAAI,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC;IAC/C,CAAC;CACF;AArCD,sCAqCC;AAED,SAAS,0BAA0B,CACjC,OAAuD;IAEvD,MAAM,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,aAAa,EAAE,EAAE;QACzD,OAAO,aAAa,KAAK,OAAO,CAAC,QAAQ,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,IAAI,SAAS,EAAE,CAAC;QACd,MAAM,IAAI,KAAK,CACb,oCAAoC,GAAG,OAAO,CAAC,QAAQ,CAAC,IAAI,CAC7D,CAAC;IACJ,CAAC;AACH,CAAC;AAED,SAAS,wBAAwB,CAC/B,OAAuD,EACvD,SAAoC;IAEpC,MAAM,mBAAmB,GAAG,OAAO,CAAC,SAAS,CAAC,MAAM,CAClD,CAAC,aAA8B,EAAE,EAAE;QACjC,OAAO,CAAC,YAAY,CAAC,aAAa,EAAE,SAAS,CAAC,CAAC;IACjD,CAAC,CACF,CAAC;IACF,IAAI,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACrC,OAAO;IACT,CAAC;IAED,MAAM,cAAc,GAAG,mBAAmB;SACvC,GAAG,CAAC,CAAC,aAAa,EAAE,EAAE,CAAC,OAAO,aAAa,CAAC,IAAI,EAAE,CAAC;SACnD,IAAI,CAAC,IAAI,CAAC,CAAC;IACd,MAAM,IAAI,KAAK,CACb,GAAG,OAAO,CAAC,QAAQ,CAAC,IAAI,uCAAuC,cAAc,EAAE,CAChF,CAAC;AACJ,CAAC;AAED,SAAS,YAAY,CACnB,GAAwB,EACxB,SAAoC;IAEpC,OAAO,SAAS,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,KAAK,GAAG,CAAC,CAAC;AAC/D,CAAC;AAED,SAAS,UAAU,CACjB,GAAkB,EAClB,OAAmC;IAEnC,OAAO,CAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,QAAQ,MAAK,GAAG,CAAC;AACnC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceProvider.d.ts","sourceRoot":"","sources":["../src/serviceProvider.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAE1C,MAAM,WAAW,eAAe;IAC9B,MAAM,CAAC,CAAC,EAAE,GAAG,EAAE,UAAU,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;CAC3C"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"serviceProvider.js","sourceRoot":"","sources":["../src/serviceProvider.ts"],"names":[],"mappings":""}
|
package/package.json
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "composed-di",
|
|
3
|
+
"private": false,
|
|
4
|
+
"version": "0.0.1",
|
|
5
|
+
"author": "Juan Herrera juanhr454@gmail.com",
|
|
6
|
+
"type": "commonjs",
|
|
7
|
+
"main": "./dist/index.js",
|
|
8
|
+
"types": "./dist/index.d.ts",
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"types": "./dist/index.d.ts",
|
|
12
|
+
"require": "./dist/index.js"
|
|
13
|
+
}
|
|
14
|
+
},
|
|
15
|
+
"files": [
|
|
16
|
+
"dist",
|
|
17
|
+
"src"
|
|
18
|
+
],
|
|
19
|
+
"sideEffects": false,
|
|
20
|
+
"keywords": [
|
|
21
|
+
"dependency injection",
|
|
22
|
+
"di"
|
|
23
|
+
],
|
|
24
|
+
"license": "MIT",
|
|
25
|
+
"repository": {
|
|
26
|
+
"type": "git",
|
|
27
|
+
"url": "https://github.com/imherrera/composed-di"
|
|
28
|
+
},
|
|
29
|
+
"scripts": {
|
|
30
|
+
"build": "tsc --build",
|
|
31
|
+
"prepublishOnly": "npm run build"
|
|
32
|
+
},
|
|
33
|
+
"devDependencies": {
|
|
34
|
+
"@tsconfig/recommended": "^1.0.10",
|
|
35
|
+
"@types/node": "24.0.10",
|
|
36
|
+
"prettier": "^3.3.1",
|
|
37
|
+
"typescript": "^5.2.2"
|
|
38
|
+
}
|
|
39
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { ServiceKey } from './serviceKey';
|
|
2
|
+
|
|
3
|
+
// Helper types to extract the type from ServiceKey
|
|
4
|
+
type ServiceType<T> = T extends ServiceKey<infer U> ? U : never;
|
|
5
|
+
|
|
6
|
+
// Helper types to convert an array of ServiceKey to tuple of their types
|
|
7
|
+
type DependencyTypes<T extends ServiceKey<unknown>[]> = {
|
|
8
|
+
[K in keyof T]: ServiceType<T[K]>;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
export interface ServiceFactory<T, D extends ServiceKey<unknown>[] = []> {
|
|
12
|
+
provides: ServiceKey<T>;
|
|
13
|
+
dependsOn: D;
|
|
14
|
+
|
|
15
|
+
initialize(...dependencies: DependencyTypes<D>): Promise<T>;
|
|
16
|
+
|
|
17
|
+
dispose(instance: T): void;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function singletonFactory<T, D extends ServiceKey<unknown>[] = []>({
|
|
21
|
+
provides,
|
|
22
|
+
dependsOn = [] as unknown as D,
|
|
23
|
+
initialize,
|
|
24
|
+
dispose = () => {},
|
|
25
|
+
}: {
|
|
26
|
+
provides: ServiceKey<T>;
|
|
27
|
+
dependsOn?: D;
|
|
28
|
+
initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;
|
|
29
|
+
dispose?: (instance: T) => void;
|
|
30
|
+
}): ServiceFactory<T, D> {
|
|
31
|
+
let instance: T | undefined;
|
|
32
|
+
|
|
33
|
+
return {
|
|
34
|
+
provides,
|
|
35
|
+
dependsOn,
|
|
36
|
+
async initialize(...dependencies: DependencyTypes<D>): Promise<T> {
|
|
37
|
+
if (instance) {
|
|
38
|
+
return instance;
|
|
39
|
+
}
|
|
40
|
+
instance = await initialize(...dependencies);
|
|
41
|
+
return instance;
|
|
42
|
+
},
|
|
43
|
+
dispose(serviceInstance: T): void {
|
|
44
|
+
if (instance === serviceInstance) {
|
|
45
|
+
dispose(serviceInstance);
|
|
46
|
+
instance = undefined;
|
|
47
|
+
}
|
|
48
|
+
},
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function oneShotFactory<T, D extends ServiceKey<unknown>[] = []>({
|
|
53
|
+
provides,
|
|
54
|
+
dependsOn,
|
|
55
|
+
initialize,
|
|
56
|
+
dispose = () => {},
|
|
57
|
+
}: {
|
|
58
|
+
provides: ServiceKey<T>;
|
|
59
|
+
dependsOn: D;
|
|
60
|
+
initialize: (...dependencies: DependencyTypes<D>) => Promise<T>;
|
|
61
|
+
dispose?: (instance: T) => void;
|
|
62
|
+
}): ServiceFactory<T, D> {
|
|
63
|
+
return {
|
|
64
|
+
provides,
|
|
65
|
+
dependsOn,
|
|
66
|
+
initialize,
|
|
67
|
+
dispose,
|
|
68
|
+
};
|
|
69
|
+
}
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import { ServiceKey } from './serviceKey';
|
|
2
|
+
import { ServiceFactory } from './serviceFactory';
|
|
3
|
+
import { ServiceProvider } from './serviceProvider';
|
|
4
|
+
|
|
5
|
+
export class ServiceModule implements ServiceProvider {
|
|
6
|
+
private readonly factories: ServiceFactory<any, any>[] = [];
|
|
7
|
+
|
|
8
|
+
constructor(factories: Set<ServiceFactory<unknown, ServiceKey<unknown>[]>>) {
|
|
9
|
+
this.factories = Array.from(factories);
|
|
10
|
+
this.factories.forEach((factory) => {
|
|
11
|
+
checkRecursiveDependencies(factory);
|
|
12
|
+
checkMissingDependencies(factory, this.factories);
|
|
13
|
+
});
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
public async inject<T>(key: ServiceKey<T>): Promise<T> {
|
|
17
|
+
const factory = this.factories.find((factory) => {
|
|
18
|
+
return isSuitable(key, factory);
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
// Check if a factory to supply the requested key was not found
|
|
22
|
+
if (!factory) {
|
|
23
|
+
throw new Error(`Could not find a suitable factory for ${key.name}`);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
// Resolve all dependencies first
|
|
27
|
+
const dependencies = await Promise.all(
|
|
28
|
+
factory.dependsOn.map((dependencyKey: ServiceKey<unknown>) => {
|
|
29
|
+
return this.inject(dependencyKey);
|
|
30
|
+
}),
|
|
31
|
+
);
|
|
32
|
+
|
|
33
|
+
// Call the factory to retrieve the dependency
|
|
34
|
+
return factory.initialize(...dependencies);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
static from(
|
|
38
|
+
factories: ServiceFactory<unknown, ServiceKey<unknown>[]>[],
|
|
39
|
+
): ServiceModule {
|
|
40
|
+
return new ServiceModule(new Set(factories));
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function checkRecursiveDependencies(
|
|
45
|
+
factory: ServiceFactory<unknown, ServiceKey<unknown>[]>,
|
|
46
|
+
) {
|
|
47
|
+
const recursive = factory.dependsOn.some((dependencyKey) => {
|
|
48
|
+
return dependencyKey === factory.provides;
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
if (recursive) {
|
|
52
|
+
throw new Error(
|
|
53
|
+
'Recursive dependency detected on: ' + factory.provides.name,
|
|
54
|
+
);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function checkMissingDependencies(
|
|
59
|
+
factory: ServiceFactory<unknown, ServiceKey<unknown>[]>,
|
|
60
|
+
factories: ServiceFactory<unknown>[],
|
|
61
|
+
) {
|
|
62
|
+
const missingDependencies = factory.dependsOn.filter(
|
|
63
|
+
(dependencyKey: ServiceKey<any>) => {
|
|
64
|
+
return !isRegistered(dependencyKey, factories);
|
|
65
|
+
},
|
|
66
|
+
);
|
|
67
|
+
if (missingDependencies.length === 0) {
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
const dependencyList = missingDependencies
|
|
72
|
+
.map((dependencyKey) => ` -> ${dependencyKey.name}`)
|
|
73
|
+
.join('\n');
|
|
74
|
+
throw new Error(
|
|
75
|
+
`${factory.provides.name} will fail because it depends on:\n ${dependencyList}`,
|
|
76
|
+
);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function isRegistered(
|
|
80
|
+
key: ServiceKey<unknown>,
|
|
81
|
+
factories: ServiceFactory<unknown>[],
|
|
82
|
+
) {
|
|
83
|
+
return factories.some((factory) => factory.provides === key);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function isSuitable<T, D extends ServiceKey<unknown>[]>(
|
|
87
|
+
key: ServiceKey<T>,
|
|
88
|
+
factory: ServiceFactory<unknown, D>,
|
|
89
|
+
): factory is ServiceFactory<T, D> {
|
|
90
|
+
return factory?.provides === key;
|
|
91
|
+
}
|