@jilimb0/tgwrapper 0.4.0
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 +38 -0
- package/dist/adapters/aws-lambda-handler.d.ts +29 -0
- package/dist/adapters/aws-lambda-handler.d.ts.map +1 -0
- package/dist/adapters/aws-lambda-handler.js +39 -0
- package/dist/adapters/aws-lambda-handler.js.map +1 -0
- package/dist/adapters/cloudflare-worker-handler.d.ts +7 -0
- package/dist/adapters/cloudflare-worker-handler.d.ts.map +1 -0
- package/dist/adapters/cloudflare-worker-handler.js +24 -0
- package/dist/adapters/cloudflare-worker-handler.js.map +1 -0
- package/dist/adapters/node-http-handler.d.ts +20 -0
- package/dist/adapters/node-http-handler.d.ts.map +1 -0
- package/dist/adapters/node-http-handler.js +41 -0
- package/dist/adapters/node-http-handler.js.map +1 -0
- package/dist/adapters/webhook-handler.d.ts +14 -0
- package/dist/adapters/webhook-handler.d.ts.map +1 -0
- package/dist/adapters/webhook-handler.js +54 -0
- package/dist/adapters/webhook-handler.js.map +1 -0
- package/dist/core/api-client.d.ts +18 -0
- package/dist/core/api-client.d.ts.map +1 -0
- package/dist/core/api-client.js +139 -0
- package/dist/core/api-client.js.map +1 -0
- package/dist/core/bot-kernel.d.ts +29 -0
- package/dist/core/bot-kernel.d.ts.map +1 -0
- package/dist/core/bot-kernel.js +99 -0
- package/dist/core/bot-kernel.js.map +1 -0
- package/dist/core/circuit-breaker.d.ts +19 -0
- package/dist/core/circuit-breaker.d.ts.map +1 -0
- package/dist/core/circuit-breaker.js +59 -0
- package/dist/core/circuit-breaker.js.map +1 -0
- package/dist/core/context.d.ts +29 -0
- package/dist/core/context.d.ts.map +1 -0
- package/dist/core/context.js +67 -0
- package/dist/core/context.js.map +1 -0
- package/dist/core/errors.d.ts +21 -0
- package/dist/core/errors.d.ts.map +1 -0
- package/dist/core/errors.js +45 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/fsm/session-manager.d.ts +22 -0
- package/dist/fsm/session-manager.d.ts.map +1 -0
- package/dist/fsm/session-manager.js +79 -0
- package/dist/fsm/session-manager.js.map +1 -0
- package/dist/guards/bounded-concurrency.d.ts +14 -0
- package/dist/guards/bounded-concurrency.d.ts.map +1 -0
- package/dist/guards/bounded-concurrency.js +48 -0
- package/dist/guards/bounded-concurrency.js.map +1 -0
- package/dist/guards/token-bucket-rate-limiter.d.ts +11 -0
- package/dist/guards/token-bucket-rate-limiter.d.ts.map +1 -0
- package/dist/guards/token-bucket-rate-limiter.js +27 -0
- package/dist/guards/token-bucket-rate-limiter.js.map +1 -0
- package/dist/index.d.ts +27 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +23 -0
- package/dist/index.js.map +1 -0
- package/dist/observability/ecs-logger.d.ts +18 -0
- package/dist/observability/ecs-logger.d.ts.map +1 -0
- package/dist/observability/ecs-logger.js +42 -0
- package/dist/observability/ecs-logger.js.map +1 -0
- package/dist/observability/metrics.d.ts +12 -0
- package/dist/observability/metrics.d.ts.map +1 -0
- package/dist/observability/metrics.js +31 -0
- package/dist/observability/metrics.js.map +1 -0
- package/dist/router/router.d.ts +30 -0
- package/dist/router/router.d.ts.map +1 -0
- package/dist/router/router.js +62 -0
- package/dist/router/router.js.map +1 -0
- package/dist/runtime/bot-runtime.d.ts +25 -0
- package/dist/runtime/bot-runtime.d.ts.map +1 -0
- package/dist/runtime/bot-runtime.js +59 -0
- package/dist/runtime/bot-runtime.js.map +1 -0
- package/dist/storage/memory-session-storage.d.ts +12 -0
- package/dist/storage/memory-session-storage.d.ts.map +1 -0
- package/dist/storage/memory-session-storage.js +53 -0
- package/dist/storage/memory-session-storage.js.map +1 -0
- package/dist/storage/redis-session-storage.d.ts +45 -0
- package/dist/storage/redis-session-storage.d.ts.map +1 -0
- package/dist/storage/redis-session-storage.js +101 -0
- package/dist/storage/redis-session-storage.js.map +1 -0
- package/dist/tenant/key-namespace.d.ts +7 -0
- package/dist/tenant/key-namespace.d.ts.map +1 -0
- package/dist/tenant/key-namespace.js +7 -0
- package/dist/tenant/key-namespace.js.map +1 -0
- package/dist/types/core.d.ts +106 -0
- package/dist/types/core.d.ts.map +1 -0
- package/dist/types/core.js +2 -0
- package/dist/types/core.js.map +1 -0
- package/dist/types/telegram.d.ts +2 -0
- package/dist/types/telegram.d.ts.map +1 -0
- package/dist/types/telegram.js +2 -0
- package/dist/types/telegram.js.map +1 -0
- package/dist/update-loop/polling.d.ts +13 -0
- package/dist/update-loop/polling.d.ts.map +1 -0
- package/dist/update-loop/polling.js +40 -0
- package/dist/update-loop/polling.js.map +1 -0
- package/dist/update-loop/update-validator.d.ts +4 -0
- package/dist/update-loop/update-validator.d.ts.map +1 -0
- package/dist/update-loop/update-validator.js +17 -0
- package/dist/update-loop/update-validator.js.map +1 -0
- package/dist/update-loop/webhook.d.ts +15 -0
- package/dist/update-loop/webhook.d.ts.map +1 -0
- package/dist/update-loop/webhook.js +30 -0
- package/dist/update-loop/webhook.js.map +1 -0
- package/package.json +55 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"circuit-breaker.d.ts","sourceRoot":"","sources":["../../src/core/circuit-breaker.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,kBAAkB,CAAC;AAE9D,KAAK,YAAY,GAAG,QAAQ,GAAG,MAAM,GAAG,WAAW,CAAC;AAQpD,qBAAa,cAAc;IACzB,OAAO,CAAC,KAAK,CAA0B;IACvC,OAAO,CAAC,QAAQ,CAAK;IACrB,OAAO,CAAC,QAAQ,CAAK;IACrB,OAAO,CAAC,gBAAgB,CAAK;IAC7B,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAwB;gBAE7B,OAAO,CAAC,EAAE,OAAO,CAAC,qBAAqB,CAAC;IAIpD,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAmBlC,SAAS,IAAI,IAAI;IAMjB,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAe9B,QAAQ,IAAI;QAAE,KAAK,EAAE,YAAY,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAA;KAAE;CAM7D"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { CircuitOpenError } from './errors.js';
|
|
2
|
+
const DEFAULT_CIRCUIT = {
|
|
3
|
+
failureThreshold: 5,
|
|
4
|
+
cooldownMs: 15_000,
|
|
5
|
+
halfOpenMaxRequests: 1
|
|
6
|
+
};
|
|
7
|
+
export class CircuitBreaker {
|
|
8
|
+
state = 'closed';
|
|
9
|
+
failures = 0;
|
|
10
|
+
openedAt = 0;
|
|
11
|
+
halfOpenRequests = 0;
|
|
12
|
+
options;
|
|
13
|
+
constructor(options) {
|
|
14
|
+
this.options = { ...DEFAULT_CIRCUIT, ...options };
|
|
15
|
+
}
|
|
16
|
+
beforeRequest(nowMs) {
|
|
17
|
+
if (this.state === 'open') {
|
|
18
|
+
const elapsed = nowMs - this.openedAt;
|
|
19
|
+
if (elapsed >= this.options.cooldownMs) {
|
|
20
|
+
this.state = 'half_open';
|
|
21
|
+
this.halfOpenRequests = 0;
|
|
22
|
+
}
|
|
23
|
+
else {
|
|
24
|
+
throw new CircuitOpenError(this.options.cooldownMs - elapsed);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
if (this.state === 'half_open') {
|
|
28
|
+
if (this.halfOpenRequests >= this.options.halfOpenMaxRequests) {
|
|
29
|
+
throw new CircuitOpenError(this.options.cooldownMs);
|
|
30
|
+
}
|
|
31
|
+
this.halfOpenRequests += 1;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
onSuccess() {
|
|
35
|
+
this.state = 'closed';
|
|
36
|
+
this.failures = 0;
|
|
37
|
+
this.halfOpenRequests = 0;
|
|
38
|
+
}
|
|
39
|
+
onFailure(nowMs) {
|
|
40
|
+
if (this.state === 'half_open') {
|
|
41
|
+
this.state = 'open';
|
|
42
|
+
this.openedAt = nowMs;
|
|
43
|
+
this.halfOpenRequests = 0;
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
this.failures += 1;
|
|
47
|
+
if (this.failures >= this.options.failureThreshold) {
|
|
48
|
+
this.state = 'open';
|
|
49
|
+
this.openedAt = nowMs;
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
snapshot() {
|
|
53
|
+
return {
|
|
54
|
+
state: this.state,
|
|
55
|
+
failures: this.failures
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=circuit-breaker.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"circuit-breaker.js","sourceRoot":"","sources":["../../src/core/circuit-breaker.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AAK/C,MAAM,eAAe,GAA0B;IAC7C,gBAAgB,EAAE,CAAC;IACnB,UAAU,EAAE,MAAM;IAClB,mBAAmB,EAAE,CAAC;CACvB,CAAC;AAEF,MAAM,OAAO,cAAc;IACjB,KAAK,GAAiB,QAAQ,CAAC;IAC/B,QAAQ,GAAG,CAAC,CAAC;IACb,QAAQ,GAAG,CAAC,CAAC;IACb,gBAAgB,GAAG,CAAC,CAAC;IACZ,OAAO,CAAwB;IAEhD,YAAmB,OAAwC;QACzD,IAAI,CAAC,OAAO,GAAG,EAAE,GAAG,eAAe,EAAE,GAAG,OAAO,EAAE,CAAC;IACpD,CAAC;IAEM,aAAa,CAAC,KAAa;QAChC,IAAI,IAAI,CAAC,KAAK,KAAK,MAAM,EAAE,CAAC;YAC1B,MAAM,OAAO,GAAG,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC;YACtC,IAAI,OAAO,IAAI,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;gBACvC,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC;gBACzB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;YAC5B,CAAC;iBAAM,CAAC;gBACN,MAAM,IAAI,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,GAAG,OAAO,CAAC,CAAC;YAChE,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,KAAK,KAAK,WAAW,EAAE,CAAC;YAC/B,IAAI,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,OAAO,CAAC,mBAAmB,EAAE,CAAC;gBAC9D,MAAM,IAAI,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;YACtD,CAAC;YACD,IAAI,CAAC,gBAAgB,IAAI,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAEM,SAAS;QACd,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC;QACtB,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC;QAClB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;IAC5B,CAAC;IAEM,SAAS,CAAC,KAAa;QAC5B,IAAI,IAAI,CAAC,KAAK,KAAK,WAAW,EAAE,CAAC;YAC/B,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC;YACpB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,IAAI,CAAC,QAAQ,IAAI,CAAC,CAAC;QACnB,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,OAAO,CAAC,gBAAgB,EAAE,CAAC;YACnD,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC;YACpB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACxB,CAAC;IACH,CAAC;IAEM,QAAQ;QACb,OAAO;YACL,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,QAAQ,EAAE,IAAI,CAAC,QAAQ;SACxB,CAAC;IACJ,CAAC;CACF"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import type { ApiClient } from './api-client.js';
|
|
2
|
+
import type { JsonObject, SessionEnvelope } from '../types/core.js';
|
|
3
|
+
import type { CallbackQuery, Message, Update } from '../types/telegram.js';
|
|
4
|
+
interface SceneController {
|
|
5
|
+
enter(nextState: string): Promise<void>;
|
|
6
|
+
leave(): Promise<void>;
|
|
7
|
+
}
|
|
8
|
+
export declare class Context<TState extends string, TData extends JsonObject> {
|
|
9
|
+
readonly update: Update;
|
|
10
|
+
readonly state: JsonObject;
|
|
11
|
+
readonly session: SessionEnvelope<TState, TData>;
|
|
12
|
+
readonly scene: SceneController;
|
|
13
|
+
private readonly apiClient;
|
|
14
|
+
constructor(params: {
|
|
15
|
+
update: Update;
|
|
16
|
+
state?: JsonObject;
|
|
17
|
+
session: SessionEnvelope<TState, TData>;
|
|
18
|
+
sceneController: SceneController;
|
|
19
|
+
apiClient: ApiClient;
|
|
20
|
+
});
|
|
21
|
+
get message(): Message.TextMessage | undefined;
|
|
22
|
+
get callbackQuery(): CallbackQuery | undefined;
|
|
23
|
+
get fromId(): number | undefined;
|
|
24
|
+
reply(text: string, extra?: JsonObject): Promise<unknown>;
|
|
25
|
+
editMessage(text: string, extra?: JsonObject): Promise<unknown>;
|
|
26
|
+
answerCallbackQuery(text?: string): Promise<unknown>;
|
|
27
|
+
}
|
|
28
|
+
export {};
|
|
29
|
+
//# sourceMappingURL=context.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/core/context.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,KAAK,EAAE,UAAU,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAC;AACpE,OAAO,KAAK,EAAE,aAAa,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AAE3E,UAAU,eAAe;IACvB,KAAK,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IACxC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;CACxB;AAED,qBAAa,OAAO,CAAC,MAAM,SAAS,MAAM,EAAE,KAAK,SAAS,UAAU;IAClE,SAAgB,MAAM,EAAE,MAAM,CAAC;IAC/B,SAAgB,KAAK,EAAE,UAAU,CAAC;IAClC,SAAgB,OAAO,EAAE,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IACxD,SAAgB,KAAK,EAAE,eAAe,CAAC;IAEvC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAY;gBAEnB,MAAM,EAAE;QACzB,MAAM,EAAE,MAAM,CAAC;QACf,KAAK,CAAC,EAAE,UAAU,CAAC;QACnB,OAAO,EAAE,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QACxC,eAAe,EAAE,eAAe,CAAC;QACjC,SAAS,EAAE,SAAS,CAAC;KACtB;IAQD,IAAW,OAAO,IAAI,OAAO,CAAC,WAAW,GAAG,SAAS,CAMpD;IAED,IAAW,aAAa,IAAI,aAAa,GAAG,SAAS,CAEpD;IAED,IAAW,MAAM,IAAI,MAAM,GAAG,SAAS,CAQtC;IAEY,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,GAAE,UAAe,GAAG,OAAO,CAAC,OAAO,CAAC;IAa7D,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,GAAE,UAAe,GAAG,OAAO,CAAC,OAAO,CAAC;IAcnE,mBAAmB,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;CAWlE"}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
export class Context {
|
|
2
|
+
update;
|
|
3
|
+
state;
|
|
4
|
+
session;
|
|
5
|
+
scene;
|
|
6
|
+
apiClient;
|
|
7
|
+
constructor(params) {
|
|
8
|
+
this.update = params.update;
|
|
9
|
+
this.state = params.state ?? {};
|
|
10
|
+
this.session = params.session;
|
|
11
|
+
this.scene = params.sceneController;
|
|
12
|
+
this.apiClient = params.apiClient;
|
|
13
|
+
}
|
|
14
|
+
get message() {
|
|
15
|
+
const maybeMessage = this.update.message;
|
|
16
|
+
if (maybeMessage && 'text' in maybeMessage) {
|
|
17
|
+
return maybeMessage;
|
|
18
|
+
}
|
|
19
|
+
return undefined;
|
|
20
|
+
}
|
|
21
|
+
get callbackQuery() {
|
|
22
|
+
return this.update.callback_query;
|
|
23
|
+
}
|
|
24
|
+
get fromId() {
|
|
25
|
+
if (this.update.message?.from) {
|
|
26
|
+
return this.update.message.from.id;
|
|
27
|
+
}
|
|
28
|
+
if (this.update.callback_query?.from) {
|
|
29
|
+
return this.update.callback_query.from.id;
|
|
30
|
+
}
|
|
31
|
+
return undefined;
|
|
32
|
+
}
|
|
33
|
+
async reply(text, extra = {}) {
|
|
34
|
+
const chatId = this.update.message?.chat.id ?? this.update.callback_query?.message?.chat.id;
|
|
35
|
+
if (!chatId) {
|
|
36
|
+
throw new Error('Cannot reply without chat id in update.');
|
|
37
|
+
}
|
|
38
|
+
return this.apiClient.callApi('sendMessage', {
|
|
39
|
+
chat_id: chatId,
|
|
40
|
+
text,
|
|
41
|
+
...extra
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
async editMessage(text, extra = {}) {
|
|
45
|
+
const callbackMessage = this.update.callback_query?.message;
|
|
46
|
+
if (!callbackMessage) {
|
|
47
|
+
throw new Error('Cannot edit message without callback_query.message.');
|
|
48
|
+
}
|
|
49
|
+
return this.apiClient.callApi('editMessageText', {
|
|
50
|
+
chat_id: callbackMessage.chat.id,
|
|
51
|
+
message_id: callbackMessage.message_id,
|
|
52
|
+
text,
|
|
53
|
+
...extra
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
async answerCallbackQuery(text) {
|
|
57
|
+
const callbackQueryId = this.update.callback_query?.id;
|
|
58
|
+
if (!callbackQueryId) {
|
|
59
|
+
throw new Error('Cannot answer callback query for non-callback update.');
|
|
60
|
+
}
|
|
61
|
+
return this.apiClient.callApi('answerCallbackQuery', {
|
|
62
|
+
callback_query_id: callbackQueryId,
|
|
63
|
+
text
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
//# sourceMappingURL=context.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/core/context.ts"],"names":[],"mappings":"AASA,MAAM,OAAO,OAAO;IACF,MAAM,CAAS;IACf,KAAK,CAAa;IAClB,OAAO,CAAiC;IACxC,KAAK,CAAkB;IAEtB,SAAS,CAAY;IAEtC,YAAmB,MAMlB;QACC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC;QAC5B,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;QAChC,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;QAC9B,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,eAAe,CAAC;QACpC,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;IACpC,CAAC;IAED,IAAW,OAAO;QAChB,MAAM,YAAY,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;QACzC,IAAI,YAAY,IAAI,MAAM,IAAI,YAAY,EAAE,CAAC;YAC3C,OAAO,YAAmC,CAAC;QAC7C,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC;IACpC,CAAC;IAED,IAAW,MAAM;QACf,IAAI,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;QACrC,CAAC;QACD,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,IAAI,EAAE,CAAC;YACrC,OAAO,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,IAAI,CAAC,EAAE,CAAC;QAC5C,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAEM,KAAK,CAAC,KAAK,CAAC,IAAY,EAAE,QAAoB,EAAE;QACrD,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,OAAO,EAAE,IAAI,CAAC,EAAE,CAAC;QAC5F,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;QAC7D,CAAC;QAED,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,aAAa,EAAE;YAC3C,OAAO,EAAE,MAAM;YACf,IAAI;YACJ,GAAG,KAAK;SACT,CAAC,CAAC;IACL,CAAC;IAEM,KAAK,CAAC,WAAW,CAAC,IAAY,EAAE,QAAoB,EAAE;QAC3D,MAAM,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,OAAO,CAAC;QAC5D,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;QACzE,CAAC;QAED,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,iBAAiB,EAAE;YAC/C,OAAO,EAAE,eAAe,CAAC,IAAI,CAAC,EAAE;YAChC,UAAU,EAAE,eAAe,CAAC,UAAU;YACtC,IAAI;YACJ,GAAG,KAAK;SACT,CAAC,CAAC;IACL,CAAC;IAEM,KAAK,CAAC,mBAAmB,CAAC,IAAa;QAC5C,MAAM,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,EAAE,CAAC;QACvD,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;QAC3E,CAAC;QAED,OAAO,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,qBAAqB,EAAE;YACnD,iBAAiB,EAAE,eAAe;YAClC,IAAI;SACL,CAAC,CAAC;IACL,CAAC;CACF"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import type { JsonObject } from '../types/core.js';
|
|
2
|
+
export type CoreErrorCode = 'CIRCUIT_OPEN' | 'TELEGRAM_API_ERROR' | 'NETWORK_ERROR' | 'VALIDATION_ERROR' | 'SESSION_CONFLICT';
|
|
3
|
+
export declare class CoreError extends Error {
|
|
4
|
+
readonly code: CoreErrorCode;
|
|
5
|
+
readonly retryable: boolean;
|
|
6
|
+
readonly details: JsonObject | undefined;
|
|
7
|
+
constructor(code: CoreErrorCode, message: string, retryable: boolean, details?: JsonObject);
|
|
8
|
+
}
|
|
9
|
+
export declare class TelegramApiError extends CoreError {
|
|
10
|
+
readonly errorCode: number;
|
|
11
|
+
readonly description: string;
|
|
12
|
+
readonly parameters: JsonObject | undefined;
|
|
13
|
+
constructor(errorCode: number, description: string, parameters?: JsonObject);
|
|
14
|
+
}
|
|
15
|
+
export declare class CircuitOpenError extends CoreError {
|
|
16
|
+
constructor(cooldownMs: number);
|
|
17
|
+
}
|
|
18
|
+
export declare class SessionConflictError extends CoreError {
|
|
19
|
+
constructor(sessionKey: string);
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=errors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errors.d.ts","sourceRoot":"","sources":["../../src/core/errors.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAEnD,MAAM,MAAM,aAAa,GACrB,cAAc,GACd,oBAAoB,GACpB,eAAe,GACf,kBAAkB,GAClB,kBAAkB,CAAC;AAEvB,qBAAa,SAAU,SAAQ,KAAK;IAClC,SAAgB,IAAI,EAAE,aAAa,CAAC;IACpC,SAAgB,SAAS,EAAE,OAAO,CAAC;IACnC,SAAgB,OAAO,EAAE,UAAU,GAAG,SAAS,CAAC;gBAE7B,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,UAAU;CAOlG;AAED,qBAAa,gBAAiB,SAAQ,SAAS;IAC7C,SAAgB,SAAS,EAAE,MAAM,CAAC;IAClC,SAAgB,WAAW,EAAE,MAAM,CAAC;IACpC,SAAgB,UAAU,EAAE,UAAU,GAAG,SAAS,CAAC;gBAEhC,SAAS,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,UAAU;CAWnF;AAED,qBAAa,gBAAiB,SAAQ,SAAS;gBAC1B,UAAU,EAAE,MAAM;CAMtC;AAED,qBAAa,oBAAqB,SAAQ,SAAS;gBAC9B,UAAU,EAAE,MAAM;CAMtC"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
export class CoreError extends Error {
|
|
2
|
+
code;
|
|
3
|
+
retryable;
|
|
4
|
+
details;
|
|
5
|
+
constructor(code, message, retryable, details) {
|
|
6
|
+
super(message);
|
|
7
|
+
this.code = code;
|
|
8
|
+
this.retryable = retryable;
|
|
9
|
+
this.details = details;
|
|
10
|
+
this.name = 'CoreError';
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
export class TelegramApiError extends CoreError {
|
|
14
|
+
errorCode;
|
|
15
|
+
description;
|
|
16
|
+
parameters;
|
|
17
|
+
constructor(errorCode, description, parameters) {
|
|
18
|
+
super('TELEGRAM_API_ERROR', `Telegram API error ${errorCode}: ${description}`, errorCode === 429, {
|
|
19
|
+
error_code: errorCode,
|
|
20
|
+
description,
|
|
21
|
+
parameters
|
|
22
|
+
});
|
|
23
|
+
this.name = 'TelegramApiError';
|
|
24
|
+
this.errorCode = errorCode;
|
|
25
|
+
this.description = description;
|
|
26
|
+
this.parameters = parameters;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
export class CircuitOpenError extends CoreError {
|
|
30
|
+
constructor(cooldownMs) {
|
|
31
|
+
super('CIRCUIT_OPEN', `Circuit breaker is open. Retry after ${cooldownMs}ms.`, true, {
|
|
32
|
+
cooldown_ms: cooldownMs
|
|
33
|
+
});
|
|
34
|
+
this.name = 'CircuitOpenError';
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
export class SessionConflictError extends CoreError {
|
|
38
|
+
constructor(sessionKey) {
|
|
39
|
+
super('SESSION_CONFLICT', `Session conflict for key ${sessionKey}`, true, {
|
|
40
|
+
session_key: sessionKey
|
|
41
|
+
});
|
|
42
|
+
this.name = 'SessionConflictError';
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
//# sourceMappingURL=errors.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/core/errors.ts"],"names":[],"mappings":"AASA,MAAM,OAAO,SAAU,SAAQ,KAAK;IAClB,IAAI,CAAgB;IACpB,SAAS,CAAU;IACnB,OAAO,CAAyB;IAEhD,YAAmB,IAAmB,EAAE,OAAe,EAAE,SAAkB,EAAE,OAAoB;QAC/F,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAC3B,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,IAAI,GAAG,WAAW,CAAC;IAC1B,CAAC;CACF;AAED,MAAM,OAAO,gBAAiB,SAAQ,SAAS;IAC7B,SAAS,CAAS;IAClB,WAAW,CAAS;IACpB,UAAU,CAAyB;IAEnD,YAAmB,SAAiB,EAAE,WAAmB,EAAE,UAAuB;QAChF,KAAK,CAAC,oBAAoB,EAAE,sBAAsB,SAAS,KAAK,WAAW,EAAE,EAAE,SAAS,KAAK,GAAG,EAAE;YAChG,UAAU,EAAE,SAAS;YACrB,WAAW;YACX,UAAU;SACX,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,GAAG,kBAAkB,CAAC;QAC/B,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAC3B,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;QAC/B,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;IAC/B,CAAC;CACF;AAED,MAAM,OAAO,gBAAiB,SAAQ,SAAS;IAC7C,YAAmB,UAAkB;QACnC,KAAK,CAAC,cAAc,EAAE,wCAAwC,UAAU,KAAK,EAAE,IAAI,EAAE;YACnF,WAAW,EAAE,UAAU;SACxB,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,GAAG,kBAAkB,CAAC;IACjC,CAAC;CACF;AAED,MAAM,OAAO,oBAAqB,SAAQ,SAAS;IACjD,YAAmB,UAAkB;QACnC,KAAK,CAAC,kBAAkB,EAAE,4BAA4B,UAAU,EAAE,EAAE,IAAI,EAAE;YACxE,WAAW,EAAE,UAAU;SACxB,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,GAAG,sBAAsB,CAAC;IACrC,CAAC;CACF"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import type { JsonObject, MetricsCollector, SessionEnvelope, SessionStorage, VersionedValue } from '../types/core.js';
|
|
2
|
+
export interface SessionManagerOptions<TState extends string, TData extends JsonObject> {
|
|
3
|
+
storage: SessionStorage<SessionEnvelope<TState, TData>>;
|
|
4
|
+
initialData: () => TData;
|
|
5
|
+
encryptionRequired?: boolean;
|
|
6
|
+
conflictRetries?: number;
|
|
7
|
+
metrics?: MetricsCollector;
|
|
8
|
+
}
|
|
9
|
+
export declare class SessionManager<TState extends string, TData extends JsonObject> {
|
|
10
|
+
private readonly storage;
|
|
11
|
+
private readonly initialData;
|
|
12
|
+
private readonly encryptionRequired;
|
|
13
|
+
private readonly conflictRetries;
|
|
14
|
+
private readonly metrics;
|
|
15
|
+
constructor(options: SessionManagerOptions<TState, TData>);
|
|
16
|
+
load(sessionKey: string): Promise<VersionedValue<SessionEnvelope<TState, TData>>>;
|
|
17
|
+
runInSession<TResult>(sessionKey: string, runner: (session: SessionEnvelope<TState, TData>) => Promise<TResult>): Promise<TResult>;
|
|
18
|
+
private createInitialSession;
|
|
19
|
+
private cloneSession;
|
|
20
|
+
private assertSecurity;
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=session-manager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-manager.d.ts","sourceRoot":"","sources":["../../src/fsm/session-manager.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,UAAU,EAAE,gBAAgB,EAAE,eAAe,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAEtH,MAAM,WAAW,qBAAqB,CAAC,MAAM,SAAS,MAAM,EAAE,KAAK,SAAS,UAAU;IACpF,OAAO,EAAE,cAAc,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC;IACxD,WAAW,EAAE,MAAM,KAAK,CAAC;IACzB,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAC7B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,OAAO,CAAC,EAAE,gBAAgB,CAAC;CAC5B;AAED,qBAAa,cAAc,CAAC,MAAM,SAAS,MAAM,EAAE,KAAK,SAAS,UAAU;IACzE,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAiD;IACzE,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAc;IAC1C,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAU;IAC7C,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAS;IACzC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAA+B;gBAEpC,OAAO,EAAE,qBAAqB,CAAC,MAAM,EAAE,KAAK,CAAC;IAQnD,IAAI,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC;IAwBjF,YAAY,CAAC,OAAO,EAC/B,UAAU,EAAE,MAAM,EAClB,MAAM,EAAE,CAAC,OAAO,EAAE,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,KAAK,OAAO,CAAC,OAAO,CAAC,GACpE,OAAO,CAAC,OAAO,CAAC;IAwBnB,OAAO,CAAC,oBAAoB;IAU5B,OAAO,CAAC,YAAY;IAUpB,OAAO,CAAC,cAAc;CAKvB"}
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { SessionConflictError } from '../core/errors.js';
|
|
2
|
+
export class SessionManager {
|
|
3
|
+
storage;
|
|
4
|
+
initialData;
|
|
5
|
+
encryptionRequired;
|
|
6
|
+
conflictRetries;
|
|
7
|
+
metrics;
|
|
8
|
+
constructor(options) {
|
|
9
|
+
this.storage = options.storage;
|
|
10
|
+
this.initialData = options.initialData;
|
|
11
|
+
this.encryptionRequired = options.encryptionRequired ?? false;
|
|
12
|
+
this.conflictRetries = options.conflictRetries ?? 3;
|
|
13
|
+
this.metrics = options.metrics;
|
|
14
|
+
}
|
|
15
|
+
async load(sessionKey) {
|
|
16
|
+
const existing = await this.storage.getWithVersion(sessionKey);
|
|
17
|
+
if (existing) {
|
|
18
|
+
this.assertSecurity(existing.value);
|
|
19
|
+
return existing;
|
|
20
|
+
}
|
|
21
|
+
const created = this.createInitialSession();
|
|
22
|
+
const saved = await this.storage.compareAndSet(sessionKey, 0, created);
|
|
23
|
+
if (!saved.ok) {
|
|
24
|
+
this.metrics?.increment('session_conflict_count');
|
|
25
|
+
const fallback = await this.storage.getWithVersion(sessionKey);
|
|
26
|
+
if (fallback) {
|
|
27
|
+
return fallback;
|
|
28
|
+
}
|
|
29
|
+
throw new SessionConflictError(sessionKey);
|
|
30
|
+
}
|
|
31
|
+
return {
|
|
32
|
+
value: created,
|
|
33
|
+
version: created.version
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
async runInSession(sessionKey, runner) {
|
|
37
|
+
for (let attempt = 1; attempt <= this.conflictRetries + 1; attempt += 1) {
|
|
38
|
+
const current = await this.load(sessionKey);
|
|
39
|
+
const mutableSession = this.cloneSession(current.value);
|
|
40
|
+
const result = await runner(mutableSession);
|
|
41
|
+
this.assertSecurity(mutableSession);
|
|
42
|
+
mutableSession.version = current.version + 1;
|
|
43
|
+
mutableSession.updated_at = new Date().toISOString();
|
|
44
|
+
const save = await this.storage.compareAndSet(sessionKey, current.version, mutableSession);
|
|
45
|
+
if (save.ok) {
|
|
46
|
+
return result;
|
|
47
|
+
}
|
|
48
|
+
this.metrics?.increment('session_conflict_count');
|
|
49
|
+
if (attempt > this.conflictRetries) {
|
|
50
|
+
throw new SessionConflictError(sessionKey);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
throw new SessionConflictError(sessionKey);
|
|
54
|
+
}
|
|
55
|
+
createInitialSession() {
|
|
56
|
+
return {
|
|
57
|
+
current_state: null,
|
|
58
|
+
data: this.initialData(),
|
|
59
|
+
version: 1,
|
|
60
|
+
encrypted: !this.encryptionRequired,
|
|
61
|
+
updated_at: new Date().toISOString()
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
cloneSession(session) {
|
|
65
|
+
return {
|
|
66
|
+
current_state: session.current_state,
|
|
67
|
+
data: { ...session.data },
|
|
68
|
+
version: session.version,
|
|
69
|
+
encrypted: session.encrypted,
|
|
70
|
+
updated_at: session.updated_at
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
assertSecurity(session) {
|
|
74
|
+
if (this.encryptionRequired && !session.encrypted) {
|
|
75
|
+
throw new Error('Session contains sensitive data and must be encrypted before saving.');
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
//# sourceMappingURL=session-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"session-manager.js","sourceRoot":"","sources":["../../src/fsm/session-manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AAWzD,MAAM,OAAO,cAAc;IACR,OAAO,CAAiD;IACxD,WAAW,CAAc;IACzB,kBAAkB,CAAU;IAC5B,eAAe,CAAS;IACxB,OAAO,CAA+B;IAEvD,YAAmB,OAA6C;QAC9D,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;QAC/B,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;QACvC,IAAI,CAAC,kBAAkB,GAAG,OAAO,CAAC,kBAAkB,IAAI,KAAK,CAAC;QAC9D,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,CAAC,CAAC;QACpD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;IACjC,CAAC;IAEM,KAAK,CAAC,IAAI,CAAC,UAAkB;QAClC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;QAC/D,IAAI,QAAQ,EAAE,CAAC;YACb,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACpC,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5C,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,UAAU,EAAE,CAAC,EAAE,OAAO,CAAC,CAAC;QACvE,IAAI,CAAC,KAAK,CAAC,EAAE,EAAE,CAAC;YACd,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,wBAAwB,CAAC,CAAC;YAClD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;YAC/D,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO,QAAQ,CAAC;YAClB,CAAC;YACD,MAAM,IAAI,oBAAoB,CAAC,UAAU,CAAC,CAAC;QAC7C,CAAC;QAED,OAAO;YACL,KAAK,EAAE,OAAO;YACd,OAAO,EAAE,OAAO,CAAC,OAAO;SACzB,CAAC;IACJ,CAAC;IAEM,KAAK,CAAC,YAAY,CACvB,UAAkB,EAClB,MAAqE;QAErE,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,IAAI,CAAC,eAAe,GAAG,CAAC,EAAE,OAAO,IAAI,CAAC,EAAE,CAAC;YACxE,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YAC5C,MAAM,cAAc,GAAG,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YACxD,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,CAAC;YAC5C,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YAEpC,cAAc,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,GAAG,CAAC,CAAC;YAC7C,cAAc,CAAC,UAAU,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YAErD,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,UAAU,EAAE,OAAO,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;YAC3F,IAAI,IAAI,CAAC,EAAE,EAAE,CAAC;gBACZ,OAAO,MAAM,CAAC;YAChB,CAAC;YAED,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,wBAAwB,CAAC,CAAC;YAClD,IAAI,OAAO,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;gBACnC,MAAM,IAAI,oBAAoB,CAAC,UAAU,CAAC,CAAC;YAC7C,CAAC;QACH,CAAC;QAED,MAAM,IAAI,oBAAoB,CAAC,UAAU,CAAC,CAAC;IAC7C,CAAC;IAEO,oBAAoB;QAC1B,OAAO;YACL,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,IAAI,CAAC,WAAW,EAAE;YACxB,OAAO,EAAE,CAAC;YACV,SAAS,EAAE,CAAC,IAAI,CAAC,kBAAkB;YACnC,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SACrC,CAAC;IACJ,CAAC;IAEO,YAAY,CAAC,OAAuC;QAC1D,OAAO;YACL,aAAa,EAAE,OAAO,CAAC,aAAa;YACpC,IAAI,EAAE,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE;YACzB,OAAO,EAAE,OAAO,CAAC,OAAO;YACxB,SAAS,EAAE,OAAO,CAAC,SAAS;YAC5B,UAAU,EAAE,OAAO,CAAC,UAAU;SAC/B,CAAC;IACJ,CAAC;IAEO,cAAc,CAAC,OAAuC;QAC5D,IAAI,IAAI,CAAC,kBAAkB,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC;YAClD,MAAM,IAAI,KAAK,CAAC,sEAAsE,CAAC,CAAC;QAC1F,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
export declare class QueueOverflowError extends Error {
|
|
2
|
+
constructor(limit: number);
|
|
3
|
+
}
|
|
4
|
+
export declare class BoundedConcurrencyQueue {
|
|
5
|
+
private running;
|
|
6
|
+
private readonly waiters;
|
|
7
|
+
private readonly maxConcurrency;
|
|
8
|
+
private readonly maxQueue;
|
|
9
|
+
constructor(maxConcurrency: number, maxQueue: number);
|
|
10
|
+
run<T>(task: () => Promise<T>): Promise<T>;
|
|
11
|
+
private acquire;
|
|
12
|
+
private release;
|
|
13
|
+
}
|
|
14
|
+
//# sourceMappingURL=bounded-concurrency.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bounded-concurrency.d.ts","sourceRoot":"","sources":["../../src/guards/bounded-concurrency.ts"],"names":[],"mappings":"AAAA,qBAAa,kBAAmB,SAAQ,KAAK;gBACxB,KAAK,EAAE,MAAM;CAIjC;AAED,qBAAa,uBAAuB;IAClC,OAAO,CAAC,OAAO,CAAK;IACpB,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAyB;IACjD,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;gBAEf,cAAc,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM;IAK9C,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;YASzC,OAAO;IAkBrB,OAAO,CAAC,OAAO;CAOhB"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
export class QueueOverflowError extends Error {
|
|
2
|
+
constructor(limit) {
|
|
3
|
+
super(`Concurrency queue is full (limit=${limit})`);
|
|
4
|
+
this.name = 'QueueOverflowError';
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
export class BoundedConcurrencyQueue {
|
|
8
|
+
running = 0;
|
|
9
|
+
waiters = [];
|
|
10
|
+
maxConcurrency;
|
|
11
|
+
maxQueue;
|
|
12
|
+
constructor(maxConcurrency, maxQueue) {
|
|
13
|
+
this.maxConcurrency = maxConcurrency;
|
|
14
|
+
this.maxQueue = maxQueue;
|
|
15
|
+
}
|
|
16
|
+
async run(task) {
|
|
17
|
+
await this.acquire();
|
|
18
|
+
try {
|
|
19
|
+
return await task();
|
|
20
|
+
}
|
|
21
|
+
finally {
|
|
22
|
+
this.release();
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
async acquire() {
|
|
26
|
+
if (this.running < this.maxConcurrency) {
|
|
27
|
+
this.running += 1;
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
if (this.waiters.length >= this.maxQueue) {
|
|
31
|
+
throw new QueueOverflowError(this.maxQueue);
|
|
32
|
+
}
|
|
33
|
+
await new Promise((resolve) => {
|
|
34
|
+
this.waiters.push(() => {
|
|
35
|
+
this.running += 1;
|
|
36
|
+
resolve();
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
}
|
|
40
|
+
release() {
|
|
41
|
+
this.running = Math.max(0, this.running - 1);
|
|
42
|
+
const next = this.waiters.shift();
|
|
43
|
+
if (next) {
|
|
44
|
+
next();
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=bounded-concurrency.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bounded-concurrency.js","sourceRoot":"","sources":["../../src/guards/bounded-concurrency.ts"],"names":[],"mappings":"AAAA,MAAM,OAAO,kBAAmB,SAAQ,KAAK;IAC3C,YAAmB,KAAa;QAC9B,KAAK,CAAC,oCAAoC,KAAK,GAAG,CAAC,CAAC;QACpD,IAAI,CAAC,IAAI,GAAG,oBAAoB,CAAC;IACnC,CAAC;CACF;AAED,MAAM,OAAO,uBAAuB;IAC1B,OAAO,GAAG,CAAC,CAAC;IACH,OAAO,GAAsB,EAAE,CAAC;IAChC,cAAc,CAAS;IACvB,QAAQ,CAAS;IAElC,YAAmB,cAAsB,EAAE,QAAgB;QACzD,IAAI,CAAC,cAAc,GAAG,cAAc,CAAC;QACrC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;IAC3B,CAAC;IAEM,KAAK,CAAC,GAAG,CAAI,IAAsB;QACxC,MAAM,IAAI,CAAC,OAAO,EAAE,CAAC;QACrB,IAAI,CAAC;YACH,OAAO,MAAM,IAAI,EAAE,CAAC;QACtB,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,OAAO,EAAE,CAAC;QACjB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,OAAO;QACnB,IAAI,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACvC,IAAI,CAAC,OAAO,IAAI,CAAC,CAAC;YAClB,OAAO;QACT,CAAC;QAED,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACzC,MAAM,IAAI,kBAAkB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC9C,CAAC;QAED,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,EAAE;YAClC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE;gBACrB,IAAI,CAAC,OAAO,IAAI,CAAC,CAAC;gBAClB,OAAO,EAAE,CAAC;YACZ,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,OAAO;QACb,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QAClC,IAAI,IAAI,EAAE,CAAC;YACT,IAAI,EAAE,CAAC;QACT,CAAC;IACH,CAAC;CACF"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export interface RateLimitConfig {
|
|
2
|
+
capacity: number;
|
|
3
|
+
refillPerSecond: number;
|
|
4
|
+
}
|
|
5
|
+
export declare class TokenBucketRateLimiter {
|
|
6
|
+
private readonly config;
|
|
7
|
+
private readonly buckets;
|
|
8
|
+
constructor(config: RateLimitConfig);
|
|
9
|
+
allow(key: string, nowMs?: number): boolean;
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=token-bucket-rate-limiter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"token-bucket-rate-limiter.d.ts","sourceRoot":"","sources":["../../src/guards/token-bucket-rate-limiter.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,eAAe;IAC9B,QAAQ,EAAE,MAAM,CAAC;IACjB,eAAe,EAAE,MAAM,CAAC;CACzB;AAOD,qBAAa,sBAAsB;IACjC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkB;IACzC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAkC;gBAEvC,MAAM,EAAE,eAAe;IAInC,KAAK,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,SAAa,GAAG,OAAO;CAsBvD"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export class TokenBucketRateLimiter {
|
|
2
|
+
config;
|
|
3
|
+
buckets = new Map();
|
|
4
|
+
constructor(config) {
|
|
5
|
+
this.config = config;
|
|
6
|
+
}
|
|
7
|
+
allow(key, nowMs = Date.now()) {
|
|
8
|
+
const state = this.buckets.get(key) ?? {
|
|
9
|
+
tokens: this.config.capacity,
|
|
10
|
+
lastRefillMs: nowMs
|
|
11
|
+
};
|
|
12
|
+
const elapsedSec = (nowMs - state.lastRefillMs) / 1000;
|
|
13
|
+
const refilled = Math.min(this.config.capacity, state.tokens + elapsedSec * this.config.refillPerSecond);
|
|
14
|
+
const next = {
|
|
15
|
+
tokens: refilled,
|
|
16
|
+
lastRefillMs: nowMs
|
|
17
|
+
};
|
|
18
|
+
if (next.tokens < 1) {
|
|
19
|
+
this.buckets.set(key, next);
|
|
20
|
+
return false;
|
|
21
|
+
}
|
|
22
|
+
next.tokens -= 1;
|
|
23
|
+
this.buckets.set(key, next);
|
|
24
|
+
return true;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=token-bucket-rate-limiter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"token-bucket-rate-limiter.js","sourceRoot":"","sources":["../../src/guards/token-bucket-rate-limiter.ts"],"names":[],"mappings":"AAUA,MAAM,OAAO,sBAAsB;IAChB,MAAM,CAAkB;IACxB,OAAO,GAAG,IAAI,GAAG,EAAuB,CAAC;IAE1D,YAAmB,MAAuB;QACxC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;IAEM,KAAK,CAAC,GAAW,EAAE,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE;QAC1C,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI;YACrC,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;YAC5B,YAAY,EAAE,KAAK;SACpB,CAAC;QAEF,MAAM,UAAU,GAAG,CAAC,KAAK,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC;QACvD,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,KAAK,CAAC,MAAM,GAAG,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACzG,MAAM,IAAI,GAAgB;YACxB,MAAM,EAAE,QAAQ;YAChB,YAAY,EAAE,KAAK;SACpB,CAAC;QAEF,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;YAC5B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC,MAAM,IAAI,CAAC,CAAC;QACjB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAC5B,OAAO,IAAI,CAAC;IACd,CAAC;CACF"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export { ApiClient } from './core/api-client.js';
|
|
2
|
+
export { BotKernel } from './core/bot-kernel.js';
|
|
3
|
+
export { Context } from './core/context.js';
|
|
4
|
+
export { CircuitBreaker } from './core/circuit-breaker.js';
|
|
5
|
+
export { CircuitOpenError, CoreError, SessionConflictError, TelegramApiError } from './core/errors.js';
|
|
6
|
+
export { TokenBucketRateLimiter } from './guards/token-bucket-rate-limiter.js';
|
|
7
|
+
export { BoundedConcurrencyQueue, QueueOverflowError } from './guards/bounded-concurrency.js';
|
|
8
|
+
export { EcsJsonLogger } from './observability/ecs-logger.js';
|
|
9
|
+
export { InMemoryMetrics } from './observability/metrics.js';
|
|
10
|
+
export { createSessionKey, createSessionNamespace } from './tenant/key-namespace.js';
|
|
11
|
+
export { SessionManager } from './fsm/session-manager.js';
|
|
12
|
+
export { TreeRouter } from './router/router.js';
|
|
13
|
+
export { BotRuntime } from './runtime/bot-runtime.js';
|
|
14
|
+
export { MemorySessionStorage } from './storage/memory-session-storage.js';
|
|
15
|
+
export { RedisSessionStorage } from './storage/redis-session-storage.js';
|
|
16
|
+
export { PollingSource } from './update-loop/polling.js';
|
|
17
|
+
export { WebhookSource } from './update-loop/webhook.js';
|
|
18
|
+
export { isFreshUpdate, isValidTelegramUpdate } from './update-loop/update-validator.js';
|
|
19
|
+
export { AwsLambdaHandler } from './adapters/aws-lambda-handler.js';
|
|
20
|
+
export { CloudflareWorkerHandler } from './adapters/cloudflare-worker-handler.js';
|
|
21
|
+
export { NodeHttpHandler } from './adapters/node-http-handler.js';
|
|
22
|
+
export { WebhookHandler } from './adapters/webhook-handler.js';
|
|
23
|
+
export type { ApiClientOptions, CasResult, CircuitBreakerOptions, ContextShortcuts, Handler, JsonObject, JsonValue, LogEvent, Logger, MetricsCollector, PollingOptions, RetryOptions, RouteCandidate, SessionEnvelope, SessionStorage, UpdateSource, VersionedValue, WebhookHandlerOptions, WebhookRequest, WebhookResponse } from './types/core.js';
|
|
24
|
+
export type { ApiGatewayV2Event, ApiGatewayV2Response } from './adapters/aws-lambda-handler.js';
|
|
25
|
+
export type { RedisLikeClient, RedisLikeTransaction, SessionCrypto } from './storage/redis-session-storage.js';
|
|
26
|
+
export type { ApiMethods, CallbackQuery, Chat, Message, Update, User } from './types/telegram.js';
|
|
27
|
+
//# 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,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,oBAAoB,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AACvG,OAAO,EAAE,sBAAsB,EAAE,MAAM,uCAAuC,CAAC;AAC/E,OAAO,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,MAAM,iCAAiC,CAAC;AAC9F,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,gBAAgB,EAAE,sBAAsB,EAAE,MAAM,2BAA2B,CAAC;AACrF,OAAO,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAC1D,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,0BAA0B,CAAC;AACtD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;AAC3E,OAAO,EAAE,mBAAmB,EAAE,MAAM,oCAAoC,CAAC;AACzE,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,qBAAqB,EAAE,MAAM,mCAAmC,CAAC;AACzF,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AACpE,OAAO,EAAE,uBAAuB,EAAE,MAAM,yCAAyC,CAAC;AAClF,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,YAAY,EACV,gBAAgB,EAChB,SAAS,EACT,qBAAqB,EACrB,gBAAgB,EAChB,OAAO,EACP,UAAU,EACV,SAAS,EACT,QAAQ,EACR,MAAM,EACN,gBAAgB,EAChB,cAAc,EACd,YAAY,EACZ,cAAc,EACd,eAAe,EACf,cAAc,EACd,YAAY,EACZ,cAAc,EACd,qBAAqB,EACrB,cAAc,EACd,eAAe,EAChB,MAAM,iBAAiB,CAAC;AAEzB,YAAY,EAAE,iBAAiB,EAAE,oBAAoB,EAAE,MAAM,kCAAkC,CAAC;AAEhG,YAAY,EAAE,eAAe,EAAE,oBAAoB,EAAE,aAAa,EAAE,MAAM,oCAAoC,CAAC;AAE/G,YAAY,EAAE,UAAU,EAAE,aAAa,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export { ApiClient } from './core/api-client.js';
|
|
2
|
+
export { BotKernel } from './core/bot-kernel.js';
|
|
3
|
+
export { Context } from './core/context.js';
|
|
4
|
+
export { CircuitBreaker } from './core/circuit-breaker.js';
|
|
5
|
+
export { CircuitOpenError, CoreError, SessionConflictError, TelegramApiError } from './core/errors.js';
|
|
6
|
+
export { TokenBucketRateLimiter } from './guards/token-bucket-rate-limiter.js';
|
|
7
|
+
export { BoundedConcurrencyQueue, QueueOverflowError } from './guards/bounded-concurrency.js';
|
|
8
|
+
export { EcsJsonLogger } from './observability/ecs-logger.js';
|
|
9
|
+
export { InMemoryMetrics } from './observability/metrics.js';
|
|
10
|
+
export { createSessionKey, createSessionNamespace } from './tenant/key-namespace.js';
|
|
11
|
+
export { SessionManager } from './fsm/session-manager.js';
|
|
12
|
+
export { TreeRouter } from './router/router.js';
|
|
13
|
+
export { BotRuntime } from './runtime/bot-runtime.js';
|
|
14
|
+
export { MemorySessionStorage } from './storage/memory-session-storage.js';
|
|
15
|
+
export { RedisSessionStorage } from './storage/redis-session-storage.js';
|
|
16
|
+
export { PollingSource } from './update-loop/polling.js';
|
|
17
|
+
export { WebhookSource } from './update-loop/webhook.js';
|
|
18
|
+
export { isFreshUpdate, isValidTelegramUpdate } from './update-loop/update-validator.js';
|
|
19
|
+
export { AwsLambdaHandler } from './adapters/aws-lambda-handler.js';
|
|
20
|
+
export { CloudflareWorkerHandler } from './adapters/cloudflare-worker-handler.js';
|
|
21
|
+
export { NodeHttpHandler } from './adapters/node-http-handler.js';
|
|
22
|
+
export { WebhookHandler } from './adapters/webhook-handler.js';
|
|
23
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,gBAAgB,EAAE,SAAS,EAAE,oBAAoB,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AACvG,OAAO,EAAE,sBAAsB,EAAE,MAAM,uCAAuC,CAAC;AAC/E,OAAO,EAAE,uBAAuB,EAAE,kBAAkB,EAAE,MAAM,iCAAiC,CAAC;AAC9F,OAAO,EAAE,aAAa,EAAE,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAE,eAAe,EAAE,MAAM,4BAA4B,CAAC;AAC7D,OAAO,EAAE,gBAAgB,EAAE,sBAAsB,EAAE,MAAM,2BAA2B,CAAC;AACrF,OAAO,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAC1D,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,0BAA0B,CAAC;AACtD,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;AAC3E,OAAO,EAAE,mBAAmB,EAAE,MAAM,oCAAoC,CAAC;AACzE,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,MAAM,0BAA0B,CAAC;AACzD,OAAO,EAAE,aAAa,EAAE,qBAAqB,EAAE,MAAM,mCAAmC,CAAC;AACzF,OAAO,EAAE,gBAAgB,EAAE,MAAM,kCAAkC,CAAC;AACpE,OAAO,EAAE,uBAAuB,EAAE,MAAM,yCAAyC,CAAC;AAClF,OAAO,EAAE,eAAe,EAAE,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { LogEvent, Logger } from '../types/core.js';
|
|
2
|
+
export interface EcsContext {
|
|
3
|
+
serviceName: string;
|
|
4
|
+
tenantId?: string;
|
|
5
|
+
botId?: string;
|
|
6
|
+
}
|
|
7
|
+
export interface LogSink {
|
|
8
|
+
write(line: string): void;
|
|
9
|
+
}
|
|
10
|
+
export declare class EcsJsonLogger implements Logger {
|
|
11
|
+
private readonly context;
|
|
12
|
+
private readonly sink;
|
|
13
|
+
constructor(context: EcsContext, sink: LogSink);
|
|
14
|
+
log(event: LogEvent): void;
|
|
15
|
+
private toFlatData;
|
|
16
|
+
private stripUndefined;
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=ecs-logger.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ecs-logger.d.ts","sourceRoot":"","sources":["../../src/observability/ecs-logger.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAc,QAAQ,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAErE,MAAM,WAAW,UAAU;IACzB,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,OAAO;IACtB,KAAK,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI,CAAC;CAC3B;AAED,qBAAa,aAAc,YAAW,MAAM;IAC1C,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAa;IACrC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAU;gBAEZ,OAAO,EAAE,UAAU,EAAE,IAAI,EAAE,OAAO;IAK9C,GAAG,CAAC,KAAK,EAAE,QAAQ,GAAG,IAAI;IAgBjC,OAAO,CAAC,UAAU;IAYlB,OAAO,CAAC,cAAc;CASvB"}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
export class EcsJsonLogger {
|
|
2
|
+
context;
|
|
3
|
+
sink;
|
|
4
|
+
constructor(context, sink) {
|
|
5
|
+
this.context = context;
|
|
6
|
+
this.sink = sink;
|
|
7
|
+
}
|
|
8
|
+
log(event) {
|
|
9
|
+
const payload = {
|
|
10
|
+
'@timestamp': event.timestamp,
|
|
11
|
+
'log.level': event.level,
|
|
12
|
+
message: event.event,
|
|
13
|
+
'service.name': this.context.serviceName,
|
|
14
|
+
'event.action': event.event,
|
|
15
|
+
'labels.tenant_id': this.context.tenantId,
|
|
16
|
+
'labels.bot_id': this.context.botId,
|
|
17
|
+
'labels.request_id': event.requestId,
|
|
18
|
+
...this.toFlatData(event.data)
|
|
19
|
+
};
|
|
20
|
+
this.sink.write(JSON.stringify(this.stripUndefined(payload)));
|
|
21
|
+
}
|
|
22
|
+
toFlatData(data) {
|
|
23
|
+
if (!data) {
|
|
24
|
+
return {};
|
|
25
|
+
}
|
|
26
|
+
const out = {};
|
|
27
|
+
for (const [key, value] of Object.entries(data)) {
|
|
28
|
+
out[`labels.${key}`] = value;
|
|
29
|
+
}
|
|
30
|
+
return out;
|
|
31
|
+
}
|
|
32
|
+
stripUndefined(input) {
|
|
33
|
+
const out = {};
|
|
34
|
+
for (const [key, value] of Object.entries(input)) {
|
|
35
|
+
if (value !== undefined) {
|
|
36
|
+
out[key] = value;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
return out;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=ecs-logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ecs-logger.js","sourceRoot":"","sources":["../../src/observability/ecs-logger.ts"],"names":[],"mappings":"AAYA,MAAM,OAAO,aAAa;IACP,OAAO,CAAa;IACpB,IAAI,CAAU;IAE/B,YAAmB,OAAmB,EAAE,IAAa;QACnD,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC;QACvB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;IACnB,CAAC;IAEM,GAAG,CAAC,KAAe;QACxB,MAAM,OAAO,GAA4B;YACvC,YAAY,EAAE,KAAK,CAAC,SAAS;YAC7B,WAAW,EAAE,KAAK,CAAC,KAAK;YACxB,OAAO,EAAE,KAAK,CAAC,KAAK;YACpB,cAAc,EAAE,IAAI,CAAC,OAAO,CAAC,WAAW;YACxC,cAAc,EAAE,KAAK,CAAC,KAAK;YAC3B,kBAAkB,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ;YACzC,eAAe,EAAE,IAAI,CAAC,OAAO,CAAC,KAAK;YACnC,mBAAmB,EAAE,KAAK,CAAC,SAAS;YACpC,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC;SAC/B,CAAC;QAEF,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IAChE,CAAC;IAEO,UAAU,CAAC,IAAiB;QAClC,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,MAAM,GAAG,GAA4B,EAAE,CAAC;QACxC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YAChD,GAAG,CAAC,UAAU,GAAG,EAAE,CAAC,GAAG,KAAK,CAAC;QAC/B,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC;IAEO,cAAc,CAAC,KAA8B;QACnD,MAAM,GAAG,GAA4B,EAAE,CAAC;QACxC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACjD,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;gBACxB,GAAG,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;YACnB,CAAC;QACH,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC;CACF"}
|