@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.
- package/.dynamo/logs/cicd-pipeline/output.log +1637 -3567
- package/.dynamo/logs/cicd-pipeline/status.json +42 -344
- package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.d.ts +110 -0
- package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.d.ts.map +1 -0
- package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.js +419 -0
- package/build/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.js.map +1 -0
- package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.d.ts +50 -0
- package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.d.ts.map +1 -0
- package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.js +3 -0
- package/build/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.js.map +1 -0
- package/build/_modules/ai/_modules/document-ai/index.d.ts +2 -0
- package/build/_modules/ai/_modules/document-ai/index.d.ts.map +1 -1
- package/build/_modules/ai/_modules/document-ai/index.js +2 -0
- package/build/_modules/ai/_modules/document-ai/index.js.map +1 -1
- package/build/_modules/ai/_services/ai-embedding-mock.service.d.ts +81 -0
- package/build/_modules/ai/_services/ai-embedding-mock.service.d.ts.map +1 -0
- package/build/_modules/ai/_services/ai-embedding-mock.service.js +167 -0
- package/build/_modules/ai/_services/ai-embedding-mock.service.js.map +1 -0
- package/build/_modules/ai/_services/ai-embedding-provider.registry.d.ts +52 -0
- package/build/_modules/ai/_services/ai-embedding-provider.registry.d.ts.map +1 -0
- package/build/_modules/ai/_services/ai-embedding-provider.registry.js +79 -0
- package/build/_modules/ai/_services/ai-embedding-provider.registry.js.map +1 -0
- package/build/_modules/ai/_services/lmstudio-embedding.control-service.d.ts +111 -0
- package/build/_modules/ai/_services/lmstudio-embedding.control-service.d.ts.map +1 -0
- package/build/_modules/ai/_services/lmstudio-embedding.control-service.js +298 -0
- package/build/_modules/ai/_services/lmstudio-embedding.control-service.js.map +1 -0
- package/build/_modules/ai/index.d.ts +5 -0
- package/build/_modules/ai/index.d.ts.map +1 -1
- package/build/_modules/ai/index.js +8 -0
- package/build/_modules/ai/index.js.map +1 -1
- package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.d.ts +59 -0
- package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.d.ts.map +1 -0
- package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.js +169 -0
- package/build/_modules/data-readers/_collections/dynts-sqlite-reader.util.js.map +1 -0
- package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.d.ts +32 -0
- package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.d.ts.map +1 -0
- package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.js +8 -0
- package/build/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.js.map +1 -0
- package/build/_modules/data-readers/index.d.ts +3 -0
- package/build/_modules/data-readers/index.d.ts.map +1 -0
- package/build/_modules/data-readers/index.js +11 -0
- package/build/_modules/data-readers/index.js.map +1 -0
- package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.d.ts +36 -0
- package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.d.ts.map +1 -0
- package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.js +54 -0
- package/build/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.js.map +1 -0
- package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.d.ts +70 -0
- package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.d.ts.map +1 -0
- package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.js +123 -0
- package/build/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.js.map +1 -0
- package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.d.ts +43 -0
- package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.d.ts.map +1 -0
- package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.js +72 -0
- package/build/_modules/local-vector-search/_services/lvs-vector-persist.data-service.js.map +1 -0
- package/build/_modules/local-vector-search/index.d.ts +3 -0
- package/build/_modules/local-vector-search/index.d.ts.map +1 -1
- package/build/_modules/local-vector-search/index.js +4 -0
- package/build/_modules/local-vector-search/index.js.map +1 -1
- package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.d.ts +109 -0
- package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.d.ts.map +1 -0
- package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.js +14 -0
- package/build/_modules/mcp/_models/interfaces/dynts-mcp.interface.js.map +1 -0
- package/build/_modules/mcp/_services/dynts-mcp-server.service-base.d.ts +71 -0
- package/build/_modules/mcp/_services/dynts-mcp-server.service-base.d.ts.map +1 -0
- package/build/_modules/mcp/_services/dynts-mcp-server.service-base.js +99 -0
- package/build/_modules/mcp/_services/dynts-mcp-server.service-base.js.map +1 -0
- package/build/_modules/mcp/_services/dynts-mcp.adapter.d.ts +57 -0
- package/build/_modules/mcp/_services/dynts-mcp.adapter.d.ts.map +1 -0
- package/build/_modules/mcp/_services/dynts-mcp.adapter.js +139 -0
- package/build/_modules/mcp/_services/dynts-mcp.adapter.js.map +1 -0
- package/build/_modules/mcp/index.d.ts +4 -0
- package/build/_modules/mcp/index.d.ts.map +1 -0
- package/build/_modules/mcp/index.js +13 -0
- package/build/_modules/mcp/index.js.map +1 -0
- package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.d.ts +19 -0
- package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.d.ts.map +1 -0
- package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.js +23 -0
- package/build/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.js.map +1 -0
- package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.d.ts +44 -0
- package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.d.ts.map +1 -0
- package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.js +68 -0
- package/build/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.js.map +1 -0
- package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.d.ts +89 -0
- package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.d.ts.map +1 -0
- package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.js +12 -0
- package/build/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.js.map +1 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.d.ts +84 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.d.ts.map +1 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.js +220 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.control-service.js.map +1 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.d.ts +54 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.d.ts.map +1 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.js +76 -0
- package/build/_modules/scoped-config/_services/dynts-scoped-config.data-service.js.map +1 -0
- package/build/_modules/scoped-config/index.d.ts +6 -0
- package/build/_modules/scoped-config/index.d.ts.map +1 -0
- package/build/_modules/scoped-config/index.js +15 -0
- package/build/_modules/scoped-config/index.js.map +1 -0
- package/package.json +58 -2
- package/pnpm-workspace.yaml +1 -0
- package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.spec.ts +295 -0
- package/src/_modules/ai/_modules/document-ai/_collections/dai-code-chunking.util.ts +552 -0
- package/src/_modules/ai/_modules/document-ai/_models/interfaces/dai-code-chunk.interface.ts +68 -0
- package/src/_modules/ai/_modules/document-ai/index.ts +2 -0
- package/src/_modules/ai/_services/ai-embedding-mock.service.spec.ts +115 -0
- package/src/_modules/ai/_services/ai-embedding-mock.service.ts +233 -0
- package/src/_modules/ai/_services/ai-embedding-provider.registry.spec.ts +110 -0
- package/src/_modules/ai/_services/ai-embedding-provider.registry.ts +114 -0
- package/src/_modules/ai/_services/lmstudio-embedding.control-service.spec.ts +197 -0
- package/src/_modules/ai/_services/lmstudio-embedding.control-service.ts +399 -0
- package/src/_modules/ai/index.ts +10 -0
- package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.spec.ts +176 -0
- package/src/_modules/data-readers/_collections/dynts-sqlite-reader.util.ts +203 -0
- package/src/_modules/data-readers/_models/interfaces/dynts-sqlite-reader.interface.ts +33 -0
- package/src/_modules/data-readers/index.ts +11 -0
- package/src/_modules/local-vector-search/_models/data-models/lvs-vector-persist.data-model.ts +60 -0
- package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.spec.ts +198 -0
- package/src/_modules/local-vector-search/_services/lvs-persistent-vector-pool.control-service.ts +150 -0
- package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.spec.ts +167 -0
- package/src/_modules/local-vector-search/_services/lvs-vector-persist.data-service.ts +108 -0
- package/src/_modules/local-vector-search/index.ts +6 -1
- package/src/_modules/mcp/_models/interfaces/dynts-mcp.interface.ts +111 -0
- package/src/_modules/mcp/_services/dynts-mcp-server.service-base.spec.ts +151 -0
- package/src/_modules/mcp/_services/dynts-mcp-server.service-base.ts +125 -0
- package/src/_modules/mcp/_services/dynts-mcp.adapter.ts +168 -0
- package/src/_modules/mcp/index.ts +13 -0
- package/src/_modules/scoped-config/_enums/dynts-scoped-config-level.enum.ts +22 -0
- package/src/_modules/scoped-config/_models/data-models/dynts-scoped-config.data-model.ts +82 -0
- package/src/_modules/scoped-config/_models/interfaces/dynts-scoped-config.interface.ts +107 -0
- package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.spec.ts +312 -0
- package/src/_modules/scoped-config/_services/dynts-scoped-config.control-service.ts +311 -0
- package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.spec.ts +123 -0
- package/src/_modules/scoped-config/_services/dynts-scoped-config.data-service.ts +108 -0
- 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';
|