@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.
Files changed (102) hide show
  1. package/README.md +38 -0
  2. package/dist/adapters/aws-lambda-handler.d.ts +29 -0
  3. package/dist/adapters/aws-lambda-handler.d.ts.map +1 -0
  4. package/dist/adapters/aws-lambda-handler.js +39 -0
  5. package/dist/adapters/aws-lambda-handler.js.map +1 -0
  6. package/dist/adapters/cloudflare-worker-handler.d.ts +7 -0
  7. package/dist/adapters/cloudflare-worker-handler.d.ts.map +1 -0
  8. package/dist/adapters/cloudflare-worker-handler.js +24 -0
  9. package/dist/adapters/cloudflare-worker-handler.js.map +1 -0
  10. package/dist/adapters/node-http-handler.d.ts +20 -0
  11. package/dist/adapters/node-http-handler.d.ts.map +1 -0
  12. package/dist/adapters/node-http-handler.js +41 -0
  13. package/dist/adapters/node-http-handler.js.map +1 -0
  14. package/dist/adapters/webhook-handler.d.ts +14 -0
  15. package/dist/adapters/webhook-handler.d.ts.map +1 -0
  16. package/dist/adapters/webhook-handler.js +54 -0
  17. package/dist/adapters/webhook-handler.js.map +1 -0
  18. package/dist/core/api-client.d.ts +18 -0
  19. package/dist/core/api-client.d.ts.map +1 -0
  20. package/dist/core/api-client.js +139 -0
  21. package/dist/core/api-client.js.map +1 -0
  22. package/dist/core/bot-kernel.d.ts +29 -0
  23. package/dist/core/bot-kernel.d.ts.map +1 -0
  24. package/dist/core/bot-kernel.js +99 -0
  25. package/dist/core/bot-kernel.js.map +1 -0
  26. package/dist/core/circuit-breaker.d.ts +19 -0
  27. package/dist/core/circuit-breaker.d.ts.map +1 -0
  28. package/dist/core/circuit-breaker.js +59 -0
  29. package/dist/core/circuit-breaker.js.map +1 -0
  30. package/dist/core/context.d.ts +29 -0
  31. package/dist/core/context.d.ts.map +1 -0
  32. package/dist/core/context.js +67 -0
  33. package/dist/core/context.js.map +1 -0
  34. package/dist/core/errors.d.ts +21 -0
  35. package/dist/core/errors.d.ts.map +1 -0
  36. package/dist/core/errors.js +45 -0
  37. package/dist/core/errors.js.map +1 -0
  38. package/dist/fsm/session-manager.d.ts +22 -0
  39. package/dist/fsm/session-manager.d.ts.map +1 -0
  40. package/dist/fsm/session-manager.js +79 -0
  41. package/dist/fsm/session-manager.js.map +1 -0
  42. package/dist/guards/bounded-concurrency.d.ts +14 -0
  43. package/dist/guards/bounded-concurrency.d.ts.map +1 -0
  44. package/dist/guards/bounded-concurrency.js +48 -0
  45. package/dist/guards/bounded-concurrency.js.map +1 -0
  46. package/dist/guards/token-bucket-rate-limiter.d.ts +11 -0
  47. package/dist/guards/token-bucket-rate-limiter.d.ts.map +1 -0
  48. package/dist/guards/token-bucket-rate-limiter.js +27 -0
  49. package/dist/guards/token-bucket-rate-limiter.js.map +1 -0
  50. package/dist/index.d.ts +27 -0
  51. package/dist/index.d.ts.map +1 -0
  52. package/dist/index.js +23 -0
  53. package/dist/index.js.map +1 -0
  54. package/dist/observability/ecs-logger.d.ts +18 -0
  55. package/dist/observability/ecs-logger.d.ts.map +1 -0
  56. package/dist/observability/ecs-logger.js +42 -0
  57. package/dist/observability/ecs-logger.js.map +1 -0
  58. package/dist/observability/metrics.d.ts +12 -0
  59. package/dist/observability/metrics.d.ts.map +1 -0
  60. package/dist/observability/metrics.js +31 -0
  61. package/dist/observability/metrics.js.map +1 -0
  62. package/dist/router/router.d.ts +30 -0
  63. package/dist/router/router.d.ts.map +1 -0
  64. package/dist/router/router.js +62 -0
  65. package/dist/router/router.js.map +1 -0
  66. package/dist/runtime/bot-runtime.d.ts +25 -0
  67. package/dist/runtime/bot-runtime.d.ts.map +1 -0
  68. package/dist/runtime/bot-runtime.js +59 -0
  69. package/dist/runtime/bot-runtime.js.map +1 -0
  70. package/dist/storage/memory-session-storage.d.ts +12 -0
  71. package/dist/storage/memory-session-storage.d.ts.map +1 -0
  72. package/dist/storage/memory-session-storage.js +53 -0
  73. package/dist/storage/memory-session-storage.js.map +1 -0
  74. package/dist/storage/redis-session-storage.d.ts +45 -0
  75. package/dist/storage/redis-session-storage.d.ts.map +1 -0
  76. package/dist/storage/redis-session-storage.js +101 -0
  77. package/dist/storage/redis-session-storage.js.map +1 -0
  78. package/dist/tenant/key-namespace.d.ts +7 -0
  79. package/dist/tenant/key-namespace.d.ts.map +1 -0
  80. package/dist/tenant/key-namespace.js +7 -0
  81. package/dist/tenant/key-namespace.js.map +1 -0
  82. package/dist/types/core.d.ts +106 -0
  83. package/dist/types/core.d.ts.map +1 -0
  84. package/dist/types/core.js +2 -0
  85. package/dist/types/core.js.map +1 -0
  86. package/dist/types/telegram.d.ts +2 -0
  87. package/dist/types/telegram.d.ts.map +1 -0
  88. package/dist/types/telegram.js +2 -0
  89. package/dist/types/telegram.js.map +1 -0
  90. package/dist/update-loop/polling.d.ts +13 -0
  91. package/dist/update-loop/polling.d.ts.map +1 -0
  92. package/dist/update-loop/polling.js +40 -0
  93. package/dist/update-loop/polling.js.map +1 -0
  94. package/dist/update-loop/update-validator.d.ts +4 -0
  95. package/dist/update-loop/update-validator.d.ts.map +1 -0
  96. package/dist/update-loop/update-validator.js +17 -0
  97. package/dist/update-loop/update-validator.js.map +1 -0
  98. package/dist/update-loop/webhook.d.ts +15 -0
  99. package/dist/update-loop/webhook.d.ts.map +1 -0
  100. package/dist/update-loop/webhook.js +30 -0
  101. package/dist/update-loop/webhook.js.map +1 -0
  102. 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":"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"}
@@ -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"}
@@ -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"}