@futdevpro/nts-dynamo 1.15.57 → 1.15.60

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 (134) hide show
  1. package/.dynamo/logs/cicd-pipeline/output.log +1637 -3567
  2. package/.dynamo/logs/cicd-pipeline/status.json +42 -344
  3. package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.d.ts +110 -0
  4. package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.d.ts.map +1 -0
  5. package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.js +419 -0
  6. package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.js.map +1 -0
  7. package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.d.ts +50 -0
  8. package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.d.ts.map +1 -0
  9. package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.js +3 -0
  10. package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.js.map +1 -0
  11. package/build/_modules/ai/_modules/document-ai/index.d.ts +2 -0
  12. package/build/_modules/ai/_modules/document-ai/index.d.ts.map +1 -1
  13. package/build/_modules/ai/_modules/document-ai/index.js +2 -0
  14. package/build/_modules/ai/_modules/document-ai/index.js.map +1 -1
  15. package/build/_modules/ai/_services/ai-embedding-mock.service.d.ts +81 -0
  16. package/build/_modules/ai/_services/ai-embedding-mock.service.d.ts.map +1 -0
  17. package/build/_modules/ai/_services/ai-embedding-mock.service.js +167 -0
  18. package/build/_modules/ai/_services/ai-embedding-mock.service.js.map +1 -0
  19. package/build/_modules/ai/_services/ai-embedding-provider.registry.d.ts +52 -0
  20. package/build/_modules/ai/_services/ai-embedding-provider.registry.d.ts.map +1 -0
  21. package/build/_modules/ai/_services/ai-embedding-provider.registry.js +79 -0
  22. package/build/_modules/ai/_services/ai-embedding-provider.registry.js.map +1 -0
  23. package/build/_modules/ai/_services/lmstudio-embedding.control-service.d.ts +111 -0
  24. package/build/_modules/ai/_services/lmstudio-embedding.control-service.d.ts.map +1 -0
  25. package/build/_modules/ai/_services/lmstudio-embedding.control-service.js +298 -0
  26. package/build/_modules/ai/_services/lmstudio-embedding.control-service.js.map +1 -0
  27. package/build/_modules/ai/index.d.ts +5 -0
  28. package/build/_modules/ai/index.d.ts.map +1 -1
  29. package/build/_modules/ai/index.js +8 -0
  30. package/build/_modules/ai/index.js.map +1 -1
  31. package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.d.ts +59 -0
  32. package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.d.ts.map +1 -0
  33. package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.js +169 -0
  34. package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.js.map +1 -0
  35. package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.d.ts +32 -0
  36. package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.d.ts.map +1 -0
  37. package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.js +8 -0
  38. package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.js.map +1 -0
  39. package/build/_modules/data-readers/index.d.ts +3 -0
  40. package/build/_modules/data-readers/index.d.ts.map +1 -0
  41. package/build/_modules/data-readers/index.js +11 -0
  42. package/build/_modules/data-readers/index.js.map +1 -0
  43. package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.d.ts +36 -0
  44. package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.d.ts.map +1 -0
  45. package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.js +54 -0
  46. package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.js.map +1 -0
  47. package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.d.ts +70 -0
  48. package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.d.ts.map +1 -0
  49. package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.js +123 -0
  50. package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.js.map +1 -0
  51. package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.d.ts +43 -0
  52. package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.d.ts.map +1 -0
  53. package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.js +72 -0
  54. package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.js.map +1 -0
  55. package/build/_modules/local-vector-search/index.d.ts +3 -0
  56. package/build/_modules/local-vector-search/index.d.ts.map +1 -1
  57. package/build/_modules/local-vector-search/index.js +4 -0
  58. package/build/_modules/local-vector-search/index.js.map +1 -1
  59. package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.d.ts +109 -0
  60. package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.d.ts.map +1 -0
  61. package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.js +14 -0
  62. package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.js.map +1 -0
  63. package/build/_modules/mcp/_services/dynts-mcp-server.service-base.d.ts +71 -0
  64. package/build/_modules/mcp/_services/dynts-mcp-server.service-base.d.ts.map +1 -0
  65. package/build/_modules/mcp/_services/dynts-mcp-server.service-base.js +99 -0
  66. package/build/_modules/mcp/_services/dynts-mcp-server.service-base.js.map +1 -0
  67. package/build/_modules/mcp/_services/dynts-mcp.adapter.d.ts +57 -0
  68. package/build/_modules/mcp/_services/dynts-mcp.adapter.d.ts.map +1 -0
  69. package/build/_modules/mcp/_services/dynts-mcp.adapter.js +139 -0
  70. package/build/_modules/mcp/_services/dynts-mcp.adapter.js.map +1 -0
  71. package/build/_modules/mcp/index.d.ts +4 -0
  72. package/build/_modules/mcp/index.d.ts.map +1 -0
  73. package/build/_modules/mcp/index.js +13 -0
  74. package/build/_modules/mcp/index.js.map +1 -0
  75. package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.d.ts +19 -0
  76. package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.d.ts.map +1 -0
  77. package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.js +23 -0
  78. package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.js.map +1 -0
  79. package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.d.ts +44 -0
  80. package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.d.ts.map +1 -0
  81. package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.js +68 -0
  82. package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.js.map +1 -0
  83. package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.d.ts +89 -0
  84. package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.d.ts.map +1 -0
  85. package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.js +12 -0
  86. package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.js.map +1 -0
  87. package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.d.ts +84 -0
  88. package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.d.ts.map +1 -0
  89. package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.js +220 -0
  90. package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.js.map +1 -0
  91. package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.d.ts +54 -0
  92. package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.d.ts.map +1 -0
  93. package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.js +76 -0
  94. package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.js.map +1 -0
  95. package/build/_modules/scoped-config/index.d.ts +6 -0
  96. package/build/_modules/scoped-config/index.d.ts.map +1 -0
  97. package/build/_modules/scoped-config/index.js +15 -0
  98. package/build/_modules/scoped-config/index.js.map +1 -0
  99. package/package.json +58 -2
  100. package/pnpm-workspace.yaml +1 -0
  101. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.spec.ts +295 -0
  102. package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.ts +552 -0
  103. package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.ts +68 -0
  104. package/src/_modules/ai/_modules/document-ai/index.ts +2 -0
  105. package/src/_modules/ai/_services/ai-embedding-mock.service.spec.ts +115 -0
  106. package/src/_modules/ai/_services/ai-embedding-mock.service.ts +233 -0
  107. package/src/_modules/ai/_services/ai-embedding-provider.registry.spec.ts +110 -0
  108. package/src/_modules/ai/_services/ai-embedding-provider.registry.ts +114 -0
  109. package/src/_modules/ai/_services/lmstudio-embedding.control-service.spec.ts +197 -0
  110. package/src/_modules/ai/_services/lmstudio-embedding.control-service.ts +399 -0
  111. package/src/_modules/ai/index.ts +10 -0
  112. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.spec.ts +176 -0
  113. package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.ts +203 -0
  114. package/src/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.ts +33 -0
  115. package/src/_modules/data-readers/index.ts +11 -0
  116. package/src/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.ts +60 -0
  117. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.spec.ts +198 -0
  118. package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.ts +150 -0
  119. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.spec.ts +167 -0
  120. package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.ts +108 -0
  121. package/src/_modules/local-vector-search/index.ts +6 -1
  122. package/src/_modules/mcp/_models/interfaces/dynts-mcp.interface.ts +111 -0
  123. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.spec.ts +151 -0
  124. package/src/_modules/mcp/_services/dynts-mcp-server.service-base.ts +125 -0
  125. package/src/_modules/mcp/_services/dynts-mcp.adapter.ts +168 -0
  126. package/src/_modules/mcp/index.ts +13 -0
  127. package/src/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.ts +22 -0
  128. package/src/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.ts +82 -0
  129. package/src/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.ts +107 -0
  130. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.spec.ts +312 -0
  131. package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.ts +311 -0
  132. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.spec.ts +123 -0
  133. package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.ts +108 -0
  134. package/src/_modules/scoped-config/index.ts +17 -0
@@ -0,0 +1,311 @@
1
+
2
+ import { DyFM_DBFilterSimple } from '@futdevpro/fsm-dynamo';
3
+
4
+ import { DyNTS_ScopedConfig_Level } from '../_enums/dynts-scoped-config-level.enum';
5
+ import { DyNTS_ScopedConfig } from '../_models/data-models/dynts-scoped-config.data-model';
6
+ import {
7
+ DyNTS_ScopedConfigResolveContext,
8
+ DyNTS_ScopedConfigResolvedFrom,
9
+ DyNTS_ScopedConfigResolvedValue,
10
+ DyNTS_ScopedConfigResolveOptions,
11
+ DyNTS_ScopedConfigSetOptions,
12
+ DyNTS_ScopedConfigValue
13
+ } from '../_models/interfaces/dynts-scoped-config.interface';
14
+ import { DyNTS_ScopedConfig_DataService } from './dynts-scoped-config.data-service';
15
+
16
+ /**
17
+ * Egy memória-cache bejegyzés (a `resolveAll` egy-Mongo-körös eredménye, rövid TTL-lel).
18
+ */
19
+ interface DyNTS_ScopedConfigCacheEntry {
20
+ /** A cache-elt aktív rekordok (egy kontextus-szeletre). */
21
+ records: DyNTS_ScopedConfig[];
22
+ /** Lejárati timestamp (epoch ms). */
23
+ expiresAt: number;
24
+ }
25
+
26
+ /**
27
+ * `DyNTS_ScopedConfig_ControlService` — a DB-backed, scoped config GENERIKUS feloldó-engine (BFR-AM-004).
28
+ *
29
+ * Singleton. Domain-agnosztikus: NINCS beépített kulcs-katalógus / default — a katalógus a FOGYASZTÓ
30
+ * dolga (a FAM `FAM_Config_ControlService` ennek vékony fogyasztója lesz). A bedrock CSAK a generikus
31
+ * precedencia-engine + a tárolás + a rövid-TTL cache.
32
+ *
33
+ * **Precedencia (a legspecifikusabb nyer):** `scope (leaf→root, a legmélyebb találat) > table > global
34
+ * > builtinDefault` (utóbbi a hívó által átadott fallback). A `resolve(key, opts)` egy kulcsot old fel;
35
+ * a `resolveAll(ctx)` az összes konfigurált kulcsot egy Mongo-körrel; a `set(level, key, value, opts)`
36
+ * archive-old-active + új aktív rekord + cache-invalidálás.
37
+ *
38
+ * **FIGYELEM (memory: dynts_dataservice_eager_resolve):** NEM tartunk élő `DyNTS_ScopedConfig_DataService`
39
+ * mezőt (eager DB-resolve a base-ctor-ban) — minden DB-művelet előtt lazy `new ...DataService`.
40
+ */
41
+ export class DyNTS_ScopedConfig_ControlService {
42
+
43
+ private static _instance: DyNTS_ScopedConfig_ControlService;
44
+
45
+ /** Default issuer a config-műveletekhez. */
46
+ private readonly issuer: string = 'DyNTS_ScopedConfig_ControlService';
47
+
48
+ /** Rövid-TTL memória-cache (kulcs: kontextus-szelet; `set` invalidálja). */
49
+ private cache: Map<string, DyNTS_ScopedConfigCacheEntry> = new Map<string, DyNTS_ScopedConfigCacheEntry>();
50
+
51
+ /** A cache TTL ms-ben (~5s; rövid, hogy a hot-path olcsó legyen, de a `set` után friss). */
52
+ private cacheTtlMs: number = 5000;
53
+
54
+ static getInstance(): DyNTS_ScopedConfig_ControlService {
55
+ if (!DyNTS_ScopedConfig_ControlService._instance) {
56
+ DyNTS_ScopedConfig_ControlService._instance = new DyNTS_ScopedConfig_ControlService();
57
+ }
58
+
59
+ return DyNTS_ScopedConfig_ControlService._instance;
60
+ }
61
+
62
+ // =========================================================================
63
+ // RESOLVE — precedencia: scope (leaf→root) > table > global > builtinDefault
64
+ // =========================================================================
65
+
66
+ /**
67
+ * Egy kulcs feloldása a legspecifikusabb beállított értékre. A precedencia:
68
+ * (1) SCOPE szint a `scopePath` LEVÉBŐL a gyökér felé (az első/legmélyebb találat nyer),
69
+ * (2) TABLE szint, (3) GLOBAL szint, (4) a hívó által átadott `builtinDefault` (ha van).
70
+ *
71
+ * A `scopePath` canonical formában érkezik (a fogyasztó read/write-path-ja oldotta fel).
72
+ */
73
+ async resolve<T = DyNTS_ScopedConfigValue>(
74
+ key: string,
75
+ options?: DyNTS_ScopedConfigResolveOptions<T>,
76
+ ): Promise<DyNTS_ScopedConfigResolvedValue<T>> {
77
+ const context: DyNTS_ScopedConfigResolveContext = {
78
+ table: options?.table,
79
+ scopePath: options?.scopePath,
80
+ };
81
+ const records: DyNTS_ScopedConfig[] = await this.loadEffectiveRecords(context);
82
+
83
+ return this.resolveFromRecords<T>(key, records, context, options?.builtinDefault);
84
+ }
85
+
86
+ /**
87
+ * A teljes effektív config egy adott kontextusra (batch). EGY Mongo-kör (cache-elt), majd memóriában
88
+ * merge: minden KONFIGURÁLT kulcsra a feloldott érték + forrás-szint. A hot-path (read/write) ezt
89
+ * használja, nem per-key kört. A builtin-only kulcsok NEM jelennek meg (nincs katalógus a bedrock-ban
90
+ * — azokat a fogyasztó iterálja a saját katalógusából a `resolve`-on át).
91
+ */
92
+ async resolveAll(context?: DyNTS_ScopedConfigResolveContext): Promise<{ [key: string]: DyNTS_ScopedConfigResolvedValue }> {
93
+ const records: DyNTS_ScopedConfig[] = await this.loadEffectiveRecords(context);
94
+ const result: { [key: string]: DyNTS_ScopedConfigResolvedValue } = {};
95
+ const keys: Set<string> = new Set<string>(records.map((record) => record.key));
96
+
97
+ for (const key of keys) {
98
+ result[key] = this.resolveFromRecords(key, records, context);
99
+ }
100
+
101
+ return result;
102
+ }
103
+
104
+ /**
105
+ * A `resolve` mag (memóriában, a betöltött rekord-halmazon). Külön metódus, hogy a `resolveAll` is
106
+ * ezt használja (egyetlen Mongo-kör után minden kulcsra). A `builtinDefault` a hívó fallback-je.
107
+ */
108
+ private resolveFromRecords<T = DyNTS_ScopedConfigValue>(
109
+ key: string,
110
+ records: DyNTS_ScopedConfig[],
111
+ context?: DyNTS_ScopedConfigResolveContext,
112
+ builtinDefault?: T,
113
+ ): DyNTS_ScopedConfigResolvedValue<T> {
114
+ const table: string | undefined = context?.table;
115
+ const scopePath = context?.scopePath ?? [];
116
+
117
+ // 1. SCOPE szint — a scopePath levéből a gyökér felé (leaf→root); az első találat nyer.
118
+ if (table && scopePath.length) {
119
+ for (let i = scopePath.length - 1; i >= 0; i--) {
120
+ const scopeId: string = scopePath[i].scopeId;
121
+ const hit: DyNTS_ScopedConfig | undefined = records.find((record) =>
122
+ record.level === DyNTS_ScopedConfig_Level.scope &&
123
+ record.tableScope === table &&
124
+ record.scopeId === scopeId &&
125
+ record.key === key,
126
+ );
127
+
128
+ if (hit) {
129
+ return this.toResolved<T>(hit, 'scope');
130
+ }
131
+ }
132
+ }
133
+
134
+ // 2. TABLE szint.
135
+ if (table) {
136
+ const hit: DyNTS_ScopedConfig | undefined = records.find((record) =>
137
+ record.level === DyNTS_ScopedConfig_Level.table &&
138
+ record.tableScope === table &&
139
+ record.key === key,
140
+ );
141
+
142
+ if (hit) {
143
+ return this.toResolved<T>(hit, 'table');
144
+ }
145
+ }
146
+
147
+ // 3. GLOBAL szint.
148
+ const globalHit: DyNTS_ScopedConfig | undefined = records.find((record) =>
149
+ record.level === DyNTS_ScopedConfig_Level.global && record.key === key,
150
+ );
151
+
152
+ if (globalHit) {
153
+ return this.toResolved<T>(globalHit, 'global');
154
+ }
155
+
156
+ // 4. builtinDefault (a hívó fallback-je; a bedrock NEM tart katalógust).
157
+ return {
158
+ value: builtinDefault,
159
+ resolvedFrom: 'builtin',
160
+ };
161
+ }
162
+
163
+ /** Egy DB-rekordot a feloldott-érték shape-re (a forrás-szinttel + audittal). */
164
+ private toResolved<T = DyNTS_ScopedConfigValue>(
165
+ record: DyNTS_ScopedConfig,
166
+ resolvedFrom: DyNTS_ScopedConfigResolvedFrom,
167
+ ): DyNTS_ScopedConfigResolvedValue<T> {
168
+ return {
169
+ value: record.value as T,
170
+ resolvedFrom: resolvedFrom,
171
+ scopeId: record.scopeId,
172
+ setBy: record.setBy,
173
+ setByDetail: record.setByDetail,
174
+ };
175
+ }
176
+
177
+ /**
178
+ * A kontextusra releváns aktív rekordok betöltése EGY Mongo-körrel, cache-elve. A filter a global +
179
+ * (ha van) table + a scopePath összes scopeId-jára szűkít — a merge memóriában fut. `set` invalidál.
180
+ */
181
+ private async loadEffectiveRecords(context?: DyNTS_ScopedConfigResolveContext): Promise<DyNTS_ScopedConfig[]> {
182
+ const cacheKey: string = this.cacheKey(context);
183
+ const cached: DyNTS_ScopedConfigCacheEntry | undefined = this.cache.get(cacheKey);
184
+
185
+ if (cached && cached.expiresAt > Date.now()) {
186
+ return cached.records;
187
+ }
188
+
189
+ const orClauses: DyFM_DBFilterSimple<DyNTS_ScopedConfig>[] = [{ level: DyNTS_ScopedConfig_Level.global }];
190
+
191
+ if (context?.table) {
192
+ orClauses.push({ level: DyNTS_ScopedConfig_Level.table, tableScope: context.table });
193
+ const scopeIds: string[] = (context.scopePath ?? []).map((ref) => ref.scopeId);
194
+
195
+ if (scopeIds.length) {
196
+ orClauses.push({ level: DyNTS_ScopedConfig_Level.scope, tableScope: context.table, scopeId: { $in: scopeIds } });
197
+ }
198
+ }
199
+
200
+ const dataService: DyNTS_ScopedConfig_DataService = new DyNTS_ScopedConfig_DataService({ issuer: this.issuer });
201
+ const records: DyNTS_ScopedConfig[] = await dataService.findActiveList({ $or: orClauses });
202
+
203
+ this.cache.set(cacheKey, { records: records, expiresAt: Date.now() + this.cacheTtlMs });
204
+
205
+ return records;
206
+ }
207
+
208
+ /** A cache-kulcs egy kontextus-szeletre (table + a scopePath scopeId-lánca). */
209
+ private cacheKey(context?: DyNTS_ScopedConfigResolveContext): string {
210
+ const table: string = context?.table ?? '-';
211
+ const scopeIds: string = (context?.scopePath ?? []).map((ref) => ref.scopeId).join('>') || '-';
212
+
213
+ return `${table}|${scopeIds}`;
214
+ }
215
+
216
+ /** A teljes cache invalidálása (`set` után). */
217
+ invalidateCache(): void {
218
+ this.cache.clear();
219
+ }
220
+
221
+ /** A cache TTL felülírása (ms); a fogyasztó hangolhatja (pl. a saját `reference.cacheTtlMs`-éből). */
222
+ setCacheTtlMs(ttlMs: number): void {
223
+ this.cacheTtlMs = ttlMs;
224
+ }
225
+
226
+ // =========================================================================
227
+ // SET — archive-old-active → write-new → invalidate cache
228
+ // =========================================================================
229
+
230
+ /**
231
+ * Egy config-érték beállítása egy adott szinten (BFR-AM-004). A `set`:
232
+ * (1) az ugyanazon `(level,tableScope,scopeId,key)` kulcson lévő AKTÍV régi rekordot soft-delete-eli
233
+ * (archív → history), (2) az új aktív rekordot kiírja (a `value` Mixed), (3) invalidálja a cache-t.
234
+ *
235
+ * **Domain-agnosztikus:** NINCS típus-/tartomány-validáció (az a fogyasztó katalógusának dolga); a
236
+ * bedrock csak a feloldási-kulcs konzisztenciáját biztosítja (table-/scope-szinten a megfelelő
237
+ * scope-azonosító jelenléte kötelező).
238
+ */
239
+ async set(
240
+ level: DyNTS_ScopedConfig_Level,
241
+ key: string,
242
+ value: unknown,
243
+ options: DyNTS_ScopedConfigSetOptions = {},
244
+ ): Promise<DyNTS_ScopedConfig> {
245
+ this.assertLevelConsistency(level, key, options);
246
+
247
+ const issuer: string = options.issuer ?? this.issuer;
248
+ const filter: DyFM_DBFilterSimple<DyNTS_ScopedConfig> =
249
+ DyNTS_ScopedConfig_DataService.levelKeyFilter(level, key, options);
250
+
251
+ // Régi aktív rekord (ha van) → soft-delete (archív; history-megőrzés).
252
+ const findService: DyNTS_ScopedConfig_DataService = new DyNTS_ScopedConfig_DataService({ issuer: issuer });
253
+ const existing: DyNTS_ScopedConfig = await findService.findActive(filter);
254
+
255
+ if (existing && existing._id) {
256
+ await findService.deleteData(existing._id);
257
+ }
258
+
259
+ // Új aktív rekord — a `value` Mixed.
260
+ const record: DyNTS_ScopedConfig = new DyNTS_ScopedConfig({
261
+ level: level,
262
+ tableScope: options.tableScope,
263
+ scopeId: options.scopeId,
264
+ key: key,
265
+ value: value,
266
+ setBy: options.setBy ?? 'system',
267
+ setByDetail: options.setByDetail,
268
+ note: options.note,
269
+ });
270
+ const writeService: DyNTS_ScopedConfig_DataService = new DyNTS_ScopedConfig_DataService({ data: record, issuer: issuer });
271
+ const saved: DyNTS_ScopedConfig = await writeService.saveData(record);
272
+
273
+ this.invalidateCache();
274
+
275
+ return saved;
276
+ }
277
+
278
+ /**
279
+ * A feloldási-kulcs minimál-konzisztenciája (NEM domain-validáció): table-/scope-szinten kötelező a
280
+ * `tableScope`, scope-szinten a `scopeId`. Ezek nélkül a feloldási-kulcs értelmetlen lenne.
281
+ */
282
+ private assertLevelConsistency(level: DyNTS_ScopedConfig_Level, key: string, options: DyNTS_ScopedConfigSetOptions): void {
283
+ if ((level === DyNTS_ScopedConfig_Level.table || level === DyNTS_ScopedConfig_Level.scope) && !options.tableScope) {
284
+ throw new Error(`DyNTS_ScopedConfig: '${key}' '${level}' szintű beállításához kötelező a tableScope.`);
285
+ }
286
+
287
+ if (level === DyNTS_ScopedConfig_Level.scope && !options.scopeId) {
288
+ throw new Error(`DyNTS_ScopedConfig: '${key}' 'scope' szintű beállításához kötelező a scopeId.`);
289
+ }
290
+ }
291
+
292
+ // =========================================================================
293
+ // HISTORY — felülírás-history (soft-delete-elt régi rekordok)
294
+ // =========================================================================
295
+
296
+ /**
297
+ * Egy feloldási-kulcs felülírás-historyja: a soft-delete-elt (archív) régi rekordok az archív
298
+ * collection-ből. Audit-trail / rollback-alap. (A FOGYASZTÓ presetjei/auditja erre épülhetnek.)
299
+ */
300
+ async getHistory(
301
+ level: DyNTS_ScopedConfig_Level,
302
+ key: string,
303
+ options: DyNTS_ScopedConfigSetOptions = {},
304
+ ): Promise<DyNTS_ScopedConfig[]> {
305
+ const filter: DyFM_DBFilterSimple<DyNTS_ScopedConfig> =
306
+ DyNTS_ScopedConfig_DataService.levelKeyFilter(level, key, options);
307
+ const dataService: DyNTS_ScopedConfig_DataService = new DyNTS_ScopedConfig_DataService({ issuer: options.issuer ?? this.issuer });
308
+
309
+ return dataService.getArchiveDataService().findDataList(filter, true);
310
+ }
311
+ }
@@ -0,0 +1,123 @@
1
+
2
+ import { DyFM_EnvironmentFlag } from '@futdevpro/fsm-dynamo';
3
+
4
+ import { DyNTS_global_settings } from '../../../_collections/global-settings.const';
5
+ import { DyNTS_GlobalService } from '../../../_services/core/global.service';
6
+ import { DyNTS_ScopedConfig_Level } from '../_enums/dynts-scoped-config-level.enum';
7
+ import { DyNTS_ScopedConfig } from '../_models/data-models/dynts-scoped-config.data-model';
8
+ import { DyNTS_ScopedConfig_DataService } from './dynts-scoped-config.data-service';
9
+
10
+ /**
11
+ * BFR-AM-004 — a `DyNTS_ScopedConfig_DataService` spec-jei. A base-ctor eager `getDBService`-jét
12
+ * stub-oljuk (memory: dynts_dataservice_eager_resolve), így NEM kell élő Mongo. A `setValue`
13
+ * spec bizonyítja a Mixed-érték ATOMIKUS `$set` írását (memory: mongoose_mixed_atomic_write).
14
+ */
15
+ describe('| DyNTS_ScopedConfig_DataService', () => {
16
+
17
+ let mockDBService: jasmine.SpyObj<{ find: () => Promise<unknown[]>; findOne: () => Promise<unknown> }>;
18
+
19
+ beforeAll(() => {
20
+ if (!DyNTS_global_settings.systemShortCodeName) {
21
+ (DyNTS_global_settings as { systemShortCodeName?: string }).systemShortCodeName = 'TEST';
22
+ }
23
+
24
+ if (!DyNTS_global_settings.env_settings) {
25
+ (DyNTS_global_settings as { env_settings?: unknown }).env_settings = {
26
+ environment: DyFM_EnvironmentFlag.local,
27
+ };
28
+ }
29
+ });
30
+
31
+ beforeEach(() => {
32
+ mockDBService = jasmine.createSpyObj('DyNTS_DBService', [ 'find', 'findOne' ]);
33
+ mockDBService.find.and.returnValue(Promise.resolve([]));
34
+ mockDBService.findOne.and.returnValue(Promise.resolve(null));
35
+ spyOn(DyNTS_GlobalService, 'getDBService').and.returnValue(mockDBService as never);
36
+ });
37
+
38
+ describe('| constructor', () => {
39
+
40
+ it('| should create the service with a data model', () => {
41
+ const data: DyNTS_ScopedConfig = new DyNTS_ScopedConfig({
42
+ level: DyNTS_ScopedConfig_Level.global,
43
+ key: 'read.topK',
44
+ value: 5,
45
+ });
46
+
47
+ const service: DyNTS_ScopedConfig_DataService =
48
+ new DyNTS_ScopedConfig_DataService({ data: data, issuer: 'issuer-123' });
49
+
50
+ expect(service).toBeInstanceOf(DyNTS_ScopedConfig_DataService);
51
+ expect(service.data).toBeDefined();
52
+ expect(service.data.key).toBe('read.topK');
53
+ expect(service.haveArchiveDataService).toBeTrue();
54
+ });
55
+
56
+ it('| should create the service without data', () => {
57
+ const service: DyNTS_ScopedConfig_DataService =
58
+ new DyNTS_ScopedConfig_DataService({ issuer: 'issuer-123' });
59
+
60
+ expect(service).toBeInstanceOf(DyNTS_ScopedConfig_DataService);
61
+ expect(service.data).toBeInstanceOf(DyNTS_ScopedConfig);
62
+ });
63
+ });
64
+
65
+ describe('| setValue — atomic $set (Mongoose Mixed)', () => {
66
+
67
+ it('| should issue an updateOne $set for the Mixed value + audit fields', async () => {
68
+ const service: DyNTS_ScopedConfig_DataService =
69
+ new DyNTS_ScopedConfig_DataService({ issuer: 'issuer-123' });
70
+
71
+ const updateSpy = spyOn(service, 'updateData').and.returnValue(Promise.resolve());
72
+
73
+ await service.setValue('rec-id', [ 'a', 'b' ], { setBy: 'cli', setByDetail: 'user-x' });
74
+
75
+ expect(updateSpy).toHaveBeenCalledTimes(1);
76
+ const arg = updateSpy.calls.mostRecent().args[0] as {
77
+ filterBy: { _id: string };
78
+ update: { $set: { value: unknown; setBy?: string; setByDetail?: string } };
79
+ };
80
+
81
+ expect(arg.filterBy._id).toBe('rec-id');
82
+ expect(arg.update.$set.value).toEqual([ 'a', 'b' ]);
83
+ expect(arg.update.$set.setBy).toBe('cli');
84
+ expect(arg.update.$set.setByDetail).toBe('user-x');
85
+ });
86
+ });
87
+
88
+ describe('| levelKeyFilter (static)', () => {
89
+
90
+ it('| should build a global filter with only level + key', () => {
91
+ const filter = DyNTS_ScopedConfig_DataService.levelKeyFilter(
92
+ DyNTS_ScopedConfig_Level.global, 'read.topK', {},
93
+ );
94
+
95
+ expect(filter).toEqual({ level: DyNTS_ScopedConfig_Level.global, key: 'read.topK' });
96
+ });
97
+
98
+ it('| should include tableScope for a table filter', () => {
99
+ const filter = DyNTS_ScopedConfig_DataService.levelKeyFilter(
100
+ DyNTS_ScopedConfig_Level.table, 'read.topK', { tableScope: 'documents' },
101
+ );
102
+
103
+ expect(filter).toEqual({
104
+ level: DyNTS_ScopedConfig_Level.table,
105
+ key: 'read.topK',
106
+ tableScope: 'documents',
107
+ });
108
+ });
109
+
110
+ it('| should include tableScope + scopeId for a scope filter', () => {
111
+ const filter = DyNTS_ScopedConfig_DataService.levelKeyFilter(
112
+ DyNTS_ScopedConfig_Level.scope, 'read.topK', { tableScope: 'documents', scopeId: 'leaf' },
113
+ );
114
+
115
+ expect(filter).toEqual({
116
+ level: DyNTS_ScopedConfig_Level.scope,
117
+ key: 'read.topK',
118
+ tableScope: 'documents',
119
+ scopeId: 'leaf',
120
+ });
121
+ });
122
+ });
123
+ });
@@ -0,0 +1,108 @@
1
+
2
+ import { DyFM_DBFilter, DyFM_DBFilterSimple } from '@futdevpro/fsm-dynamo';
3
+
4
+ import { DyNTS_ArchiveDataService } from '../../../_services/base/archive-data.service';
5
+ import { DyNTS_DataService } from '../../../_services/base/data.service';
6
+ import { DyNTS_ScopedConfig_Level } from '../_enums/dynts-scoped-config-level.enum';
7
+ import {
8
+ DyNTS_ScopedConfig,
9
+ DyNTS_scopedConfig_dataParams
10
+ } from '../_models/data-models/dynts-scoped-config.data-model';
11
+
12
+ /**
13
+ * `DyNTS_ScopedConfig_DataService` — a `dynts_scoped_config` collection CRUD + soft-delete history-rétege
14
+ * (BFR-AM-004). `extends DyNTS_DataService`; `addArchive: true` → a felülírt (régi) értékek
15
+ * soft-delete-elve megőrződnek (audit-trail). A nts-konvenció szerinti `{ data?, issuer }` constructor.
16
+ *
17
+ * **FIGYELEM (memory: dynts_dataservice_eager_resolve):** a `DyNTS_DataService` base-ctor EAGER
18
+ * `getDBService`-t hív → a control-service NEM tarthat élő data-service-példányt mezőként; minden
19
+ * művelet előtt lazy `new DyNTS_ScopedConfig_DataService(...)` kell.
20
+ *
21
+ * **A `value` írása `$set`-tel atomikus** (Mongoose Mixed silent-drop ellen — memory:
22
+ * mongoose_mixed_atomic_write): a `findOne→mutate→save` SILENT-DROP-ot ad a Mixed mezőkre, ezért az
23
+ * érték-frissítést a `setValue` `updateOne({ ... }, { $set: { value } })`-vel végzi.
24
+ */
25
+ export class DyNTS_ScopedConfig_DataService extends DyNTS_DataService<DyNTS_ScopedConfig> {
26
+
27
+ constructor(
28
+ set: {
29
+ data?: DyNTS_ScopedConfig,
30
+ issuer: string,
31
+ },
32
+ ) {
33
+ super(
34
+ set.data instanceof DyNTS_ScopedConfig ? set.data : new DyNTS_ScopedConfig(set.data),
35
+ DyNTS_scopedConfig_dataParams,
36
+ set.issuer,
37
+ );
38
+ }
39
+
40
+ /**
41
+ * Az archív (soft-delete-elt) rekordokhoz tartozó data-service (a base override-ja). A
42
+ * `DyNTS_DataService.deleteData` ezen át archiválja a felülírt aktív rekordokat (history).
43
+ */
44
+ override getArchiveDataService(): DyNTS_ArchiveDataService<DyNTS_ScopedConfig> {
45
+ return new DyNTS_ArchiveDataService<DyNTS_ScopedConfig>(
46
+ this.data,
47
+ DyNTS_scopedConfig_dataParams,
48
+ this.issuer,
49
+ );
50
+ }
51
+
52
+ /**
53
+ * Egy adott feloldási-kulcson (`level`+`tableScope`+`scopeId`+`key`) az AKTÍV (nem soft-delete-elt)
54
+ * rekord. Az aktív rekordon a `_deleted` Date soft-delete marker nincs beállítva.
55
+ */
56
+ async findActive(filter: DyFM_DBFilter<DyNTS_ScopedConfig>): Promise<DyNTS_ScopedConfig> {
57
+ return this.findData(filter, true);
58
+ }
59
+
60
+ /**
61
+ * Az adott kontextusra konfigurált ÖSSZES aktív rekord (a `resolveAll` egy Mongo-köre).
62
+ */
63
+ async findActiveList(filter: DyFM_DBFilter<DyNTS_ScopedConfig>): Promise<DyNTS_ScopedConfig[]> {
64
+ return this.findDataList(filter, true);
65
+ }
66
+
67
+ /**
68
+ * Egy létező aktív rekord `value`-jának ATOMIKUS frissítése (`$set`). A Mixed mező `findOne→mutate→
69
+ * save` útján SILENT-DROP-ot adna; a `$set` ezt elkerüli (memory: mongoose_mixed_atomic_write). A
70
+ * `setBy`/`setByDetail`/`note` audit-mezők is frissülnek.
71
+ */
72
+ async setValue(
73
+ id: string,
74
+ value: unknown,
75
+ audit: { setBy?: string, setByDetail?: string, note?: string } = {},
76
+ ): Promise<void> {
77
+ await this.updateData({
78
+ filterBy: { _id: id },
79
+ update: {
80
+ $set: {
81
+ value: value,
82
+ setBy: audit.setBy,
83
+ setByDetail: audit.setByDetail,
84
+ note: audit.note,
85
+ },
86
+ },
87
+ });
88
+ }
89
+
90
+ /** A `(level, tableScope, scopeId, key)` feloldási-kulcs filtere (a logikai unique-constraint). */
91
+ static levelKeyFilter(
92
+ level: DyNTS_ScopedConfig_Level,
93
+ key: string,
94
+ set: { tableScope?: string, scopeId?: string },
95
+ ): DyFM_DBFilterSimple<DyNTS_ScopedConfig> {
96
+ const filter: DyFM_DBFilterSimple<DyNTS_ScopedConfig> = { level: level, key: key };
97
+
98
+ if (level === DyNTS_ScopedConfig_Level.table || level === DyNTS_ScopedConfig_Level.scope) {
99
+ filter.tableScope = set.tableScope;
100
+ }
101
+
102
+ if (level === DyNTS_ScopedConfig_Level.scope) {
103
+ filter.scopeId = set.scopeId;
104
+ }
105
+
106
+ return filter;
107
+ }
108
+ }
@@ -0,0 +1,17 @@
1
+
2
+
3
+ // SCOPED-CONFIG MODULE (BFR-AM-004) — DB-backed, domain-agnostic scoped config precedence engine.
4
+ // Precedencia: scope (leaf→root) > table > global > builtinDefault.
5
+
6
+ // ENUMS
7
+ export * from './_enums/dynts-scoped-config-level.enum';
8
+
9
+ // INTERFACES
10
+ export * from './_models/interfaces/dynts-scoped-config.interface';
11
+
12
+ // DATA-MODELS
13
+ export * from './_models/data-models/dynts-scoped-config.data-model';
14
+
15
+ // SERVICES
16
+ export * from './_services/dynts-scoped-config.data-service';
17
+ export * from './_services/dynts-scoped-config.control-service';