@idevconn/ai-chat-be 0.1.2

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 (173) hide show
  1. package/dist/ai-chat.module.d.ts +6 -0
  2. package/dist/ai-chat.module.d.ts.map +1 -0
  3. package/dist/ai-chat.module.js +109 -0
  4. package/dist/ai-chat.module.js.map +1 -0
  5. package/dist/audit/audit.service.d.ts +39 -0
  6. package/dist/audit/audit.service.d.ts.map +1 -0
  7. package/dist/audit/audit.service.js +139 -0
  8. package/dist/audit/audit.service.js.map +1 -0
  9. package/dist/chat/chat.controller.d.ts +20 -0
  10. package/dist/chat/chat.controller.d.ts.map +1 -0
  11. package/dist/chat/chat.controller.js +104 -0
  12. package/dist/chat/chat.controller.js.map +1 -0
  13. package/dist/chat/chat.service.d.ts +15 -0
  14. package/dist/chat/chat.service.d.ts.map +1 -0
  15. package/dist/chat/chat.service.js +95 -0
  16. package/dist/chat/chat.service.js.map +1 -0
  17. package/dist/guards/api-key.guard.d.ts +8 -0
  18. package/dist/guards/api-key.guard.d.ts.map +1 -0
  19. package/dist/guards/api-key.guard.js +44 -0
  20. package/dist/guards/api-key.guard.js.map +1 -0
  21. package/dist/guards/llm-injection-classifier.d.ts +17 -0
  22. package/dist/guards/llm-injection-classifier.d.ts.map +1 -0
  23. package/dist/guards/llm-injection-classifier.js +82 -0
  24. package/dist/guards/llm-injection-classifier.js.map +1 -0
  25. package/dist/guards/pii.guard.d.ts +14 -0
  26. package/dist/guards/pii.guard.d.ts.map +1 -0
  27. package/dist/guards/pii.guard.js +67 -0
  28. package/dist/guards/pii.guard.js.map +1 -0
  29. package/dist/guards/prompt-injection.guard.d.ts +9 -0
  30. package/dist/guards/prompt-injection.guard.d.ts.map +1 -0
  31. package/dist/guards/prompt-injection.guard.js +157 -0
  32. package/dist/guards/prompt-injection.guard.js.map +1 -0
  33. package/dist/guards/rate-limit.guard.d.ts +8 -0
  34. package/dist/guards/rate-limit.guard.d.ts.map +1 -0
  35. package/dist/guards/rate-limit.guard.js +59 -0
  36. package/dist/guards/rate-limit.guard.js.map +1 -0
  37. package/dist/guards/topic-relevance.guard.d.ts +21 -0
  38. package/dist/guards/topic-relevance.guard.d.ts.map +1 -0
  39. package/dist/guards/topic-relevance.guard.js +59 -0
  40. package/dist/guards/topic-relevance.guard.js.map +1 -0
  41. package/dist/index.d.ts +22 -0
  42. package/dist/index.d.ts.map +1 -0
  43. package/dist/index.js +31 -0
  44. package/dist/index.js.map +1 -0
  45. package/dist/intent/embedding-intent-router.d.ts +23 -0
  46. package/dist/intent/embedding-intent-router.d.ts.map +1 -0
  47. package/dist/intent/embedding-intent-router.js +103 -0
  48. package/dist/intent/embedding-intent-router.js.map +1 -0
  49. package/dist/intent/intent-router.interface.d.ts +8 -0
  50. package/dist/intent/intent-router.interface.d.ts.map +1 -0
  51. package/dist/intent/intent-router.interface.js +5 -0
  52. package/dist/intent/intent-router.interface.js.map +1 -0
  53. package/dist/intent/intent.types.d.ts +7 -0
  54. package/dist/intent/intent.types.d.ts.map +1 -0
  55. package/dist/intent/intent.types.js +3 -0
  56. package/dist/intent/intent.types.js.map +1 -0
  57. package/dist/orchestrator/default.orchestrator.d.ts +36 -0
  58. package/dist/orchestrator/default.orchestrator.d.ts.map +1 -0
  59. package/dist/orchestrator/default.orchestrator.js +231 -0
  60. package/dist/orchestrator/default.orchestrator.js.map +1 -0
  61. package/dist/orchestrator/input-sanitizer.d.ts +8 -0
  62. package/dist/orchestrator/input-sanitizer.d.ts.map +1 -0
  63. package/dist/orchestrator/input-sanitizer.js +36 -0
  64. package/dist/orchestrator/input-sanitizer.js.map +1 -0
  65. package/dist/orchestrator/orchestrator.interface.d.ts +6 -0
  66. package/dist/orchestrator/orchestrator.interface.d.ts.map +1 -0
  67. package/dist/orchestrator/orchestrator.interface.js +5 -0
  68. package/dist/orchestrator/orchestrator.interface.js.map +1 -0
  69. package/dist/orchestrator/prompt-builder.d.ts +12 -0
  70. package/dist/orchestrator/prompt-builder.d.ts.map +1 -0
  71. package/dist/orchestrator/prompt-builder.js +55 -0
  72. package/dist/orchestrator/prompt-builder.js.map +1 -0
  73. package/dist/orchestrator/prompt-normalizer.d.ts +9 -0
  74. package/dist/orchestrator/prompt-normalizer.d.ts.map +1 -0
  75. package/dist/orchestrator/prompt-normalizer.js +77 -0
  76. package/dist/orchestrator/prompt-normalizer.js.map +1 -0
  77. package/dist/orchestrator/response-sanitizer.d.ts +26 -0
  78. package/dist/orchestrator/response-sanitizer.d.ts.map +1 -0
  79. package/dist/orchestrator/response-sanitizer.js +127 -0
  80. package/dist/orchestrator/response-sanitizer.js.map +1 -0
  81. package/dist/orchestrator/response-validator.d.ts +10 -0
  82. package/dist/orchestrator/response-validator.d.ts.map +1 -0
  83. package/dist/orchestrator/response-validator.js +31 -0
  84. package/dist/orchestrator/response-validator.js.map +1 -0
  85. package/dist/providers/gemini.provider.d.ts +13 -0
  86. package/dist/providers/gemini.provider.d.ts.map +1 -0
  87. package/dist/providers/gemini.provider.js +132 -0
  88. package/dist/providers/gemini.provider.js.map +1 -0
  89. package/dist/providers/grok.provider.d.ts +11 -0
  90. package/dist/providers/grok.provider.d.ts.map +1 -0
  91. package/dist/providers/grok.provider.js +69 -0
  92. package/dist/providers/grok.provider.js.map +1 -0
  93. package/dist/providers/openai.provider.d.ts +12 -0
  94. package/dist/providers/openai.provider.d.ts.map +1 -0
  95. package/dist/providers/openai.provider.js +87 -0
  96. package/dist/providers/openai.provider.js.map +1 -0
  97. package/dist/providers/provider-registry.d.ts +17 -0
  98. package/dist/providers/provider-registry.d.ts.map +1 -0
  99. package/dist/providers/provider-registry.js +42 -0
  100. package/dist/providers/provider-registry.js.map +1 -0
  101. package/dist/providers/provider-settings.d.ts +12 -0
  102. package/dist/providers/provider-settings.d.ts.map +1 -0
  103. package/dist/providers/provider-settings.js +3 -0
  104. package/dist/providers/provider-settings.js.map +1 -0
  105. package/dist/providers/provider.factory.d.ts +18 -0
  106. package/dist/providers/provider.factory.d.ts.map +1 -0
  107. package/dist/providers/provider.factory.js +107 -0
  108. package/dist/providers/provider.factory.js.map +1 -0
  109. package/dist/rag/chunker.d.ts +7 -0
  110. package/dist/rag/chunker.d.ts.map +1 -0
  111. package/dist/rag/chunker.js +32 -0
  112. package/dist/rag/chunker.js.map +1 -0
  113. package/dist/rag/rag.service.d.ts +20 -0
  114. package/dist/rag/rag.service.d.ts.map +1 -0
  115. package/dist/rag/rag.service.js +171 -0
  116. package/dist/rag/rag.service.js.map +1 -0
  117. package/dist/session/session.service.d.ts +32 -0
  118. package/dist/session/session.service.d.ts.map +1 -0
  119. package/dist/session/session.service.js +161 -0
  120. package/dist/session/session.service.js.map +1 -0
  121. package/dist/types/chat.types.d.ts +31 -0
  122. package/dist/types/chat.types.d.ts.map +1 -0
  123. package/dist/types/chat.types.js +3 -0
  124. package/dist/types/chat.types.js.map +1 -0
  125. package/dist/types/config.types.d.ts +139 -0
  126. package/dist/types/config.types.d.ts.map +1 -0
  127. package/dist/types/config.types.js +5 -0
  128. package/dist/types/config.types.js.map +1 -0
  129. package/dist/types/index.d.ts +4 -0
  130. package/dist/types/index.d.ts.map +1 -0
  131. package/dist/types/index.js +20 -0
  132. package/dist/types/index.js.map +1 -0
  133. package/dist/types/provider.types.d.ts +15 -0
  134. package/dist/types/provider.types.d.ts.map +1 -0
  135. package/dist/types/provider.types.js +3 -0
  136. package/dist/types/provider.types.js.map +1 -0
  137. package/dist/utils/cosine-similarity.d.ts +2 -0
  138. package/dist/utils/cosine-similarity.d.ts.map +1 -0
  139. package/dist/utils/cosine-similarity.js +21 -0
  140. package/dist/utils/cosine-similarity.js.map +1 -0
  141. package/dist/utils/error-sanitizer.d.ts +8 -0
  142. package/dist/utils/error-sanitizer.d.ts.map +1 -0
  143. package/dist/utils/error-sanitizer.js +44 -0
  144. package/dist/utils/error-sanitizer.js.map +1 -0
  145. package/dist/utils/index.d.ts +6 -0
  146. package/dist/utils/index.d.ts.map +1 -0
  147. package/dist/utils/index.js +22 -0
  148. package/dist/utils/index.js.map +1 -0
  149. package/dist/utils/logger.d.ts +11 -0
  150. package/dist/utils/logger.d.ts.map +1 -0
  151. package/dist/utils/logger.js +47 -0
  152. package/dist/utils/logger.js.map +1 -0
  153. package/dist/utils/pii-detector.d.ts +8 -0
  154. package/dist/utils/pii-detector.d.ts.map +1 -0
  155. package/dist/utils/pii-detector.js +82 -0
  156. package/dist/utils/pii-detector.js.map +1 -0
  157. package/dist/utils/retry.d.ts +8 -0
  158. package/dist/utils/retry.d.ts.map +1 -0
  159. package/dist/utils/retry.js +33 -0
  160. package/dist/utils/retry.js.map +1 -0
  161. package/dist/utils/session-token.d.ts +8 -0
  162. package/dist/utils/session-token.d.ts.map +1 -0
  163. package/dist/utils/session-token.js +46 -0
  164. package/dist/utils/session-token.js.map +1 -0
  165. package/dist/utils/sse-parser.d.ts +5 -0
  166. package/dist/utils/sse-parser.d.ts.map +1 -0
  167. package/dist/utils/sse-parser.js +45 -0
  168. package/dist/utils/sse-parser.js.map +1 -0
  169. package/dist/utils/text.d.ts +4 -0
  170. package/dist/utils/text.d.ts.map +1 -0
  171. package/dist/utils/text.js +25 -0
  172. package/dist/utils/text.js.map +1 -0
  173. package/package.json +62 -0
@@ -0,0 +1,8 @@
1
+ import { PiiType } from '../types';
2
+ export interface PiiDetectionResult {
3
+ hasPii: boolean;
4
+ types: PiiType[];
5
+ redacted: string;
6
+ }
7
+ export declare function detectPii(text: string, enabledTypes?: PiiType[]): PiiDetectionResult;
8
+ //# sourceMappingURL=pii-detector.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pii-detector.d.ts","sourceRoot":"","sources":["../../src/utils/pii-detector.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,UAAU,CAAC;AAEnC,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,OAAO,CAAC;IAChB,KAAK,EAAE,OAAO,EAAE,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAgCD,wBAAgB,SAAS,CACvB,IAAI,EAAE,MAAM,EACZ,YAAY,GAAE,OAAO,EAAkB,GACtC,kBAAkB,CAqDpB"}
@@ -0,0 +1,82 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.detectPii = detectPii;
4
+ const EMAIL_REGEX = /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/g;
5
+ const PHONE_REGEX = /(?:\+?\d{1,3}[\s\-.])?(?:\(?\d{2,4}\)?[\s\-.]?)?\d{3,4}[\s\-.]?\d{3,4}/g;
6
+ const SSN_REGEX = /\b\d{3}-\d{2}-\d{4}\b/g;
7
+ const IPV4_REGEX = /\b(?:\d{1,3}\.){3}\d{1,3}\b/g;
8
+ const CREDIT_CARD_REGEX = /\b(?:\d[ -]?){13,19}\b/g;
9
+ const DEFAULT_TYPES = ['email', 'phone', 'creditCard', 'ssn', 'ipv4'];
10
+ function luhnValid(text) {
11
+ const digits = text.replace(/\D/g, '');
12
+ if (digits.length < 13 || digits.length > 19)
13
+ return false;
14
+ let sum = 0;
15
+ let alt = false;
16
+ for (let i = digits.length - 1; i >= 0; i--) {
17
+ let d = digits.charCodeAt(i) - 48;
18
+ if (alt) {
19
+ d *= 2;
20
+ if (d > 9)
21
+ d -= 9;
22
+ }
23
+ sum += d;
24
+ alt = !alt;
25
+ }
26
+ return sum % 10 === 0;
27
+ }
28
+ function looksLikePhone(match) {
29
+ const digits = match.replace(/\D/g, '');
30
+ return digits.length >= 7 && digits.length <= 15;
31
+ }
32
+ function detectPii(text, enabledTypes = DEFAULT_TYPES) {
33
+ const types = [];
34
+ let redacted = text;
35
+ if (enabledTypes.includes('email')) {
36
+ if (EMAIL_REGEX.test(text)) {
37
+ types.push('email');
38
+ redacted = redacted.replace(EMAIL_REGEX, '[REDACTED_EMAIL]');
39
+ }
40
+ EMAIL_REGEX.lastIndex = 0;
41
+ }
42
+ if (enabledTypes.includes('ssn')) {
43
+ if (SSN_REGEX.test(text)) {
44
+ types.push('ssn');
45
+ redacted = redacted.replace(SSN_REGEX, '[REDACTED_SSN]');
46
+ }
47
+ SSN_REGEX.lastIndex = 0;
48
+ }
49
+ if (enabledTypes.includes('creditCard')) {
50
+ let foundCard = false;
51
+ redacted = redacted.replace(CREDIT_CARD_REGEX, (match) => {
52
+ if (luhnValid(match)) {
53
+ foundCard = true;
54
+ return '[REDACTED_CARD]';
55
+ }
56
+ return match;
57
+ });
58
+ if (foundCard)
59
+ types.push('creditCard');
60
+ }
61
+ if (enabledTypes.includes('ipv4')) {
62
+ if (IPV4_REGEX.test(redacted)) {
63
+ types.push('ipv4');
64
+ redacted = redacted.replace(IPV4_REGEX, '[REDACTED_IP]');
65
+ }
66
+ IPV4_REGEX.lastIndex = 0;
67
+ }
68
+ if (enabledTypes.includes('phone')) {
69
+ let foundPhone = false;
70
+ redacted = redacted.replace(PHONE_REGEX, (match) => {
71
+ if (looksLikePhone(match)) {
72
+ foundPhone = true;
73
+ return '[REDACTED_PHONE]';
74
+ }
75
+ return match;
76
+ });
77
+ if (foundPhone)
78
+ types.push('phone');
79
+ }
80
+ return { hasPii: types.length > 0, types, redacted };
81
+ }
82
+ //# sourceMappingURL=pii-detector.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pii-detector.js","sourceRoot":"","sources":["../../src/utils/pii-detector.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export interface RetryOptions {
2
+ maxAttempts: number;
3
+ baseDelayMs: number;
4
+ maxDelayMs: number;
5
+ shouldRetry?: (error: unknown) => boolean;
6
+ }
7
+ export declare function withRetry<T>(fn: () => Promise<T>, options?: Partial<RetryOptions>): Promise<T>;
8
+ //# sourceMappingURL=retry.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retry.d.ts","sourceRoot":"","sources":["../../src/utils/retry.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,YAAY;IAC3B,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,KAAK,OAAO,CAAC;CAC3C;AAQD,wBAAsB,SAAS,CAAC,CAAC,EAC/B,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,EACpB,OAAO,GAAE,OAAO,CAAC,YAAY,CAAM,GAClC,OAAO,CAAC,CAAC,CAAC,CAwBZ"}
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.withRetry = withRetry;
4
+ const DEFAULT_OPTIONS = {
5
+ maxAttempts: 3,
6
+ baseDelayMs: 1000,
7
+ maxDelayMs: 10000,
8
+ };
9
+ async function withRetry(fn, options = {}) {
10
+ const opts = { ...DEFAULT_OPTIONS, ...options };
11
+ let lastError;
12
+ for (let attempt = 0; attempt < opts.maxAttempts; attempt++) {
13
+ try {
14
+ return await fn();
15
+ }
16
+ catch (error) {
17
+ lastError = error;
18
+ if (opts.shouldRetry && !opts.shouldRetry(error)) {
19
+ throw error;
20
+ }
21
+ if (attempt === opts.maxAttempts - 1) {
22
+ break;
23
+ }
24
+ const delay = Math.min(opts.baseDelayMs * Math.pow(2, attempt), opts.maxDelayMs);
25
+ await sleep(delay);
26
+ }
27
+ }
28
+ throw lastError;
29
+ }
30
+ function sleep(ms) {
31
+ return new Promise((resolve) => setTimeout(resolve, ms));
32
+ }
33
+ //# sourceMappingURL=retry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retry.js","sourceRoot":"","sources":["../../src/utils/retry.ts"],"names":[],"mappings":";;AAaA,8BA2BC;AAjCD,MAAM,eAAe,GAAiB;IACpC,WAAW,EAAE,CAAC;IACd,WAAW,EAAE,IAAI;IACjB,UAAU,EAAE,KAAK;CAClB,CAAC;AAEK,KAAK,UAAU,SAAS,CAC7B,EAAoB,EACpB,UAAiC,EAAE;IAEnC,MAAM,IAAI,GAAG,EAAE,GAAG,eAAe,EAAE,GAAG,OAAO,EAAE,CAAC;IAChD,IAAI,SAAkB,CAAC;IAEvB,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC,WAAW,EAAE,OAAO,EAAE,EAAE,CAAC;QAC5D,IAAI,CAAC;YACH,OAAO,MAAM,EAAE,EAAE,CAAC;QACpB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,SAAS,GAAG,KAAK,CAAC;YAElB,IAAI,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC;gBACjD,MAAM,KAAK,CAAC;YACd,CAAC;YAED,IAAI,OAAO,KAAK,IAAI,CAAC,WAAW,GAAG,CAAC,EAAE,CAAC;gBACrC,MAAM;YACR,CAAC;YAED,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YACjF,MAAM,KAAK,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC;IACH,CAAC;IAED,MAAM,SAAS,CAAC;AAClB,CAAC;AAED,SAAS,KAAK,CAAC,EAAU;IACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AAC3D,CAAC"}
@@ -0,0 +1,8 @@
1
+ export declare function signSessionId(rawId: string, secret: string): string;
2
+ export interface VerifyResult {
3
+ valid: boolean;
4
+ rawId?: string;
5
+ legacy?: boolean;
6
+ }
7
+ export declare function verifySessionId(token: string, secret: string, allowLegacy?: boolean): VerifyResult;
8
+ //# sourceMappingURL=session-token.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-token.d.ts","sourceRoot":"","sources":["../../src/utils/session-token.ts"],"names":[],"mappings":"AASA,wBAAgB,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,CAInE;AAED,MAAM,WAAW,YAAY;IAC3B,KAAK,EAAE,OAAO,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAOD,wBAAgB,eAAe,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,WAAW,UAAQ,GAAG,YAAY,CA2BhG"}
@@ -0,0 +1,46 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.signSessionId = signSessionId;
4
+ exports.verifySessionId = verifySessionId;
5
+ const crypto_1 = require("crypto");
6
+ const SEPARATOR = '.';
7
+ const SIG_BYTES = 16;
8
+ function signSessionId(rawId, secret) {
9
+ const sig = (0, crypto_1.createHmac)('sha256', secret).update(rawId).digest();
10
+ const truncated = sig.subarray(0, SIG_BYTES);
11
+ return `${rawId}${SEPARATOR}${toBase64Url(truncated)}`;
12
+ }
13
+ function verifySessionId(token, secret, allowLegacy = false) {
14
+ if (!token)
15
+ return { valid: false };
16
+ const lastSep = token.lastIndexOf(SEPARATOR);
17
+ if (lastSep < 0) {
18
+ return allowLegacy ? { valid: true, rawId: token, legacy: true } : { valid: false };
19
+ }
20
+ const rawId = token.slice(0, lastSep);
21
+ const sig = token.slice(lastSep + 1);
22
+ if (!rawId || !sig)
23
+ return { valid: false };
24
+ const expected = (0, crypto_1.createHmac)('sha256', secret).update(rawId).digest().subarray(0, SIG_BYTES);
25
+ let provided;
26
+ try {
27
+ provided = fromBase64Url(sig);
28
+ }
29
+ catch {
30
+ return { valid: false };
31
+ }
32
+ if (provided.length !== expected.length)
33
+ return { valid: false };
34
+ if (!(0, crypto_1.timingSafeEqual)(provided, expected))
35
+ return { valid: false };
36
+ return { valid: true, rawId };
37
+ }
38
+ function toBase64Url(buf) {
39
+ return buf.toString('base64').replace(/=+$/g, '').replace(/\+/g, '-').replace(/\//g, '_');
40
+ }
41
+ function fromBase64Url(s) {
42
+ const padded = s.replace(/-/g, '+').replace(/_/g, '/');
43
+ const padLen = (4 - (padded.length % 4)) % 4;
44
+ return Buffer.from(padded + '='.repeat(padLen), 'base64');
45
+ }
46
+ //# sourceMappingURL=session-token.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-token.js","sourceRoot":"","sources":["../../src/utils/session-token.ts"],"names":[],"mappings":";;AASA,sCAIC;AAaD,0CA2BC;AArDD,mCAAqD;AAErD,MAAM,SAAS,GAAG,GAAG,CAAC;AACtB,MAAM,SAAS,GAAG,EAAE,CAAC;AAMrB,SAAgB,aAAa,CAAC,KAAa,EAAE,MAAc;IACzD,MAAM,GAAG,GAAG,IAAA,mBAAU,EAAC,QAAQ,EAAE,MAAM,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC;IAChE,MAAM,SAAS,GAAG,GAAG,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;IAC7C,OAAO,GAAG,KAAK,GAAG,SAAS,GAAG,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;AACzD,CAAC;AAaD,SAAgB,eAAe,CAAC,KAAa,EAAE,MAAc,EAAE,WAAW,GAAG,KAAK;IAChF,IAAI,CAAC,KAAK;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IAEpC,MAAM,OAAO,GAAG,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC;IAC7C,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;QAChB,OAAO,WAAW,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IACtF,CAAC;IAED,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IACtC,MAAM,GAAG,GAAG,KAAK,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,CAAC,CAAC;IAErC,IAAI,CAAC,KAAK,IAAI,CAAC,GAAG;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IAE5C,MAAM,QAAQ,GAAG,IAAA,mBAAU,EAAC,QAAQ,EAAE,MAAM,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;IAE5F,IAAI,QAAgB,CAAC;IACrB,IAAI,CAAC;QACH,QAAQ,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC;IAChC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IAC1B,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,QAAQ,CAAC,MAAM;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IAEjE,IAAI,CAAC,IAAA,wBAAe,EAAC,QAAQ,EAAE,QAAQ,CAAC;QAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC;IAElE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC;AAChC,CAAC;AAED,SAAS,WAAW,CAAC,GAAW;IAC9B,OAAO,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;AAC5F,CAAC;AAED,SAAS,aAAa,CAAC,CAAS;IAC9B,MAAM,MAAM,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IACvD,MAAM,MAAM,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;IAC7C,OAAO,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,QAAQ,CAAC,CAAC;AAC5D,CAAC"}
@@ -0,0 +1,5 @@
1
+ import { ChatStreamEvent } from '../types';
2
+ export declare function parseSseEvent(line: string): ChatStreamEvent | null;
3
+ export declare function formatSseEvent(event: ChatStreamEvent): string;
4
+ export declare function readSseStream(reader: ReadableStreamDefaultReader<Uint8Array>): AsyncGenerator<ChatStreamEvent>;
5
+ //# sourceMappingURL=sse-parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sse-parser.d.ts","sourceRoot":"","sources":["../../src/utils/sse-parser.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,UAAU,CAAC;AAK3C,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,eAAe,GAAG,IAAI,CAalE;AAKD,wBAAgB,cAAc,CAAC,KAAK,EAAE,eAAe,GAAG,MAAM,CAE7D;AAKD,wBAAuB,aAAa,CAClC,MAAM,EAAE,2BAA2B,CAAC,UAAU,CAAC,GAC9C,cAAc,CAAC,eAAe,CAAC,CAsBjC"}
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.parseSseEvent = parseSseEvent;
4
+ exports.formatSseEvent = formatSseEvent;
5
+ exports.readSseStream = readSseStream;
6
+ function parseSseEvent(line) {
7
+ if (!line.startsWith('data: '))
8
+ return null;
9
+ const data = line.slice(6).trim();
10
+ if (data === '[DONE]') {
11
+ return { type: 'done' };
12
+ }
13
+ try {
14
+ return JSON.parse(data);
15
+ }
16
+ catch {
17
+ return null;
18
+ }
19
+ }
20
+ function formatSseEvent(event) {
21
+ return `data: ${JSON.stringify(event)}\n\n`;
22
+ }
23
+ async function* readSseStream(reader) {
24
+ const decoder = new TextDecoder();
25
+ let buffer = '';
26
+ while (true) {
27
+ const { done, value } = await reader.read();
28
+ if (done)
29
+ break;
30
+ buffer += decoder.decode(value, { stream: true });
31
+ const lines = buffer.split('\n');
32
+ buffer = lines.pop() || '';
33
+ for (const line of lines) {
34
+ const event = parseSseEvent(line);
35
+ if (event)
36
+ yield event;
37
+ }
38
+ }
39
+ if (buffer.trim()) {
40
+ const event = parseSseEvent(buffer);
41
+ if (event)
42
+ yield event;
43
+ }
44
+ }
45
+ //# sourceMappingURL=sse-parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sse-parser.js","sourceRoot":"","sources":["../../src/utils/sse-parser.ts"],"names":[],"mappings":";;AAKA,sCAaC;AAKD,wCAEC;AAKD,sCAwBC;AAjDD,SAAgB,aAAa,CAAC,IAAY;IACxC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;QAAE,OAAO,IAAI,CAAC;IAE5C,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;IAClC,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;QACtB,OAAO,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC;IAC1B,CAAC;IAED,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,CAAoB,CAAC;IAC7C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAKD,SAAgB,cAAc,CAAC,KAAsB;IACnD,OAAO,SAAS,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC;AAC9C,CAAC;AAKM,KAAK,SAAS,CAAC,CAAC,aAAa,CAClC,MAA+C;IAE/C,MAAM,OAAO,GAAG,IAAI,WAAW,EAAE,CAAC;IAClC,IAAI,MAAM,GAAG,EAAE,CAAC;IAEhB,OAAO,IAAI,EAAE,CAAC;QACZ,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;QAC5C,IAAI,IAAI;YAAE,MAAM;QAEhB,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;QAClD,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACjC,MAAM,GAAG,KAAK,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC;QAE3B,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,MAAM,KAAK,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;YAClC,IAAI,KAAK;gBAAE,MAAM,KAAK,CAAC;QACzB,CAAC;IACH,CAAC;IAED,IAAI,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;QAClB,MAAM,KAAK,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;QACpC,IAAI,KAAK;YAAE,MAAM,KAAK,CAAC;IACzB,CAAC;AACH,CAAC"}
@@ -0,0 +1,4 @@
1
+ export declare function sanitizeInput(text: string, maxLength?: number): string;
2
+ export declare function estimateTokens(text: string): number;
3
+ export declare function truncateToTokenBudget(text: string, maxTokens: number): string;
4
+ //# sourceMappingURL=text.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"text.d.ts","sourceRoot":"","sources":["../../src/utils/text.ts"],"names":[],"mappings":"AAGA,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,SAAO,GAAG,MAAM,CAapE;AAKD,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAEnD;AAKD,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,MAAM,CAI7E"}
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.sanitizeInput = sanitizeInput;
4
+ exports.estimateTokens = estimateTokens;
5
+ exports.truncateToTokenBudget = truncateToTokenBudget;
6
+ function sanitizeInput(text, maxLength = 4000) {
7
+ let result = text.normalize('NFC');
8
+ result = result.replace(/[\x00-\x08\x0B\x0C\x0E-\x1F\x7F]/g, '');
9
+ result = result.replace(/\n{3,}/g, '\n\n');
10
+ result = result.trim();
11
+ if (result.length > maxLength) {
12
+ result = result.slice(0, maxLength);
13
+ }
14
+ return result;
15
+ }
16
+ function estimateTokens(text) {
17
+ return Math.ceil(text.length / 4);
18
+ }
19
+ function truncateToTokenBudget(text, maxTokens) {
20
+ const maxChars = maxTokens * 4;
21
+ if (text.length <= maxChars)
22
+ return text;
23
+ return text.slice(0, maxChars) + '...';
24
+ }
25
+ //# sourceMappingURL=text.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"text.js","sourceRoot":"","sources":["../../src/utils/text.ts"],"names":[],"mappings":";;AAGA,sCAaC;AAKD,wCAEC;AAKD,sDAIC;AA7BD,SAAgB,aAAa,CAAC,IAAY,EAAE,SAAS,GAAG,IAAI;IAC1D,IAAI,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAGnC,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,mCAAmC,EAAE,EAAE,CAAC,CAAC;IAEjE,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;IAE3C,MAAM,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;IACvB,IAAI,MAAM,CAAC,MAAM,GAAG,SAAS,EAAE,CAAC;QAC9B,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;IACtC,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAKD,SAAgB,cAAc,CAAC,IAAY;IACzC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;AACpC,CAAC;AAKD,SAAgB,qBAAqB,CAAC,IAAY,EAAE,SAAiB;IACnE,MAAM,QAAQ,GAAG,SAAS,GAAG,CAAC,CAAC;IAC/B,IAAI,IAAI,CAAC,MAAM,IAAI,QAAQ;QAAE,OAAO,IAAI,CAAC;IACzC,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,GAAG,KAAK,CAAC;AACzC,CAAC"}
package/package.json ADDED
@@ -0,0 +1,62 @@
1
+ {
2
+ "name": "@idevconn/ai-chat-be",
3
+ "version": "0.1.2",
4
+ "description": "Drop-in NestJS module for AI chat with multi-provider LLM, RAG, and SSE streaming",
5
+ "main": "./dist/index.js",
6
+ "types": "./dist/index.d.ts",
7
+ "files": [
8
+ "dist"
9
+ ],
10
+ "scripts": {
11
+ "build": "tsc -p tsconfig.build.json",
12
+ "typecheck": "tsc --noEmit",
13
+ "test": "vitest run --passWithNoTests",
14
+ "test:watch": "vitest",
15
+ "test:coverage": "vitest run --coverage"
16
+ },
17
+ "peerDependencies": {
18
+ "@nestjs/common": "^10.0.0 || ^11.0.0",
19
+ "@nestjs/core": "^10.0.0 || ^11.0.0",
20
+ "@nestjs/throttler": "^5.0.0 || ^6.0.0",
21
+ "reflect-metadata": "^0.1.13 || ^0.2.0",
22
+ "rxjs": "^7.0.0"
23
+ },
24
+ "peerDependenciesMeta": {
25
+ "@nestjs/throttler": {
26
+ "optional": false
27
+ }
28
+ },
29
+ "dependencies": {
30
+ "@google/genai": "^1.0.0",
31
+ "class-transformer": "^0.5.0",
32
+ "class-validator": "^0.14.0",
33
+ "openai": "^4.0.0"
34
+ },
35
+ "devDependencies": {
36
+ "@nestjs/common": "^11.0.0",
37
+ "@nestjs/core": "^11.0.0",
38
+ "@nestjs/platform-express": "^11.0.0",
39
+ "@nestjs/testing": "^11.0.0",
40
+ "@nestjs/throttler": "^6.5.0",
41
+ "@types/express": "^5.0.0",
42
+ "formdata-node": "^4.4.1",
43
+ "node-fetch": "^2.7.0",
44
+ "reflect-metadata": "^0.2.0",
45
+ "rxjs": "^7.8.0",
46
+ "typescript": "^5.7.0",
47
+ "vitest": "^3.0.0",
48
+ "whatwg-url": "^5.0.0"
49
+ },
50
+ "publishConfig": {
51
+ "access": "public"
52
+ },
53
+ "repository": {
54
+ "type": "git",
55
+ "url": "git+https://github.com/iDEVconn/ai-chat.git"
56
+ },
57
+ "bugs": {
58
+ "url": "https://github.com/iDEVconn/ai-chat/issues"
59
+ },
60
+ "homepage": "https://github.com/iDEVconn/ai-chat#readme",
61
+ "license": "MIT"
62
+ }